C++控制台:解析 METAR 数据

C++ Console : Parsing METAR data

本文关键字:METAR 数据 解析 控制台 C++      更新时间:2023-10-16

我正在开发我的第一个Web应用程序(天气可视化(,需要在后端使用一些轻量级的c ++。我正在使用wget下载原始文本,并使用c ++控制台来解析数据,然后它编写HTML。到目前为止,这很好用。

METAR基本上是来自站点的原始天气数据。(时间,日期,条件,温度等(。我目前使用的那个是:

2018/08/10 08:09
KBAZ 100809Z AUTO 00000KT 10SM BKN012 26/23 A3002 RMK AO2 T02610233

我已经能够将每组数据存储到不同的变量中。我正在看的问题集是上面的"26/23",这是以摄氏度为单位的温度和露点。

到目前为止,我有一个名为 tempAndDewpoint 的字符串,其中存储了"26/23"......我正在使用 substr(0,2( 在一个名为 temperature的新字符串中返回刚好温度。(因为第一个数字是温度(。这很好用。

我的问题是,如果温度低于 10,比如 9,会发生什么?我不能再使用子字符串(0,2(,因为这将返回"9/"作为当前温度。

我希望找到一些指导,对我来说不会太复杂,无法复制。我什至不确定如何命名这个问题,因为我不确定这个问题叫什么。肯定是常见的吧?

注意:METAR中的负温度以M为前缀。所以这些是有效的温度组:5/M2 或 M8/M12(负露点实际上是结冰点(。所以我不会在这里使用自定义解析器:

struct TTD {
short int t;
short int td;
bool parse(const char *tempAndDewpoint) {
const char *next;
t = parse_partial(tempAndDewpoint, &next);
if (*next != '/') return false;
td = parse_partial(next + 1, &next);
return (*next == '');
}
private:
static short int parse_partial(const char *beg, const char **next) {
bool neg = false;
short int val = 0;
if (*beg == 'M') {
neg = true;
beg += 1;
}
while (*beg >= '0' && *beg <= '9') {
val = val * 10 + (*beg -  '0');
beg += 1;
}
*next = beg;
if (neg) val = -val;
return val;
}
};

简单的解决方案是根本不存储为字符串。将字符串拆分为两个独立的数字。如另一个答案中所述,您确实需要注意"M"是负数的前缀,但没有读取来手动解析数字:

int parseNum(const std::string& str)
{
size_t pos;
int num;
if (!str.empty() && str.front() == 'M')
{
num = -std::stoi(str.substr(1), &pos);
if (pos != str.size() - 1)
{
throw std::invalid_argument("invalid input");
}
}
else
{
num = std::stoi(str, &pos);
if (pos != str.size())
{
throw std::invalid_argument("invalid input");
}
}
return num;
}
size_t slash = tempAndDewpoint.find("/");
if (slash == std::string::npos)
{
throw std::invalid_argument("invalid input");
}
int temp = parseNum(tempAndDewpoint.substr(0, slash));
int dew = parseNum(tempAndDewpoint.substr(slash + 1));