在对象级别或应用程序级别使用 setlocale

Using setlocale at object level or at application level

本文关键字:setlocale 对象 应用程序      更新时间:2023-10-16

我有一个设置为某个语言环境的Linux系统,它运行着一个c ++应用程序。我可以从 c++ 应用程序或操作系统本身进行std::setlocale(LC_NUMERIC, "en_US.UTF-8")(通过 /etc/default/locale 更改(。我无法访问主功能,所以我在我的函数中做了std::setlocale(LC_NUMERIC, "en_US.UTF-8"),它工作正常。

但是,我想在我的应用程序中在对象级别或某个全局级别(一次(执行此操作。我在互联网上看了很多,但没有找到太多帮助,所以决定问一下。这是我的GPS位置结构

所以,基本上

    //struct Position : std::numpunct<char> {
    struct Position {
    Position() { isValid = PR_FALSE; lat = 0; lng = 0; elevation = 0; };
    PRBool isValid;
    double lat;
    double lng;
    double elevation;
    //char do_decimal_point()   const { return '.'; }  // separate with slash
};

然后,在我的职能中,我做

mycheckposition(){
Position checkPosition;
Position mPosition;
// get some data in above.
double distance;
GetDistanceBetweenWGS84Coords(checkPosition, mPosition, distance);
}

我的问题是我应该在哪里有效安全地设置区域设置。我的位置值得到逗号而不是点,然后 GetDistanceBetweenWGS84Coords 失败。我系统上的LC_NUMERIC表示小数数字的逗号。也许,如果有人想看到GetDistance功能,它就在这里。

int GetDistanceBetweenWGS84Coords(const Position& from, const 
Position& to, double& distance)
{
const double EARTH_RADIUS_IN_METERS = 6372797.560856;
const double DEG_TO_RAD = 0.017453292519943295769236907684886;
double latitudeArc  = (from.lat - to.lat) * DEG_TO_RAD;
double longitudeArc = (from.lng - to.lng) * DEG_TO_RAD;
double latitudeH = sin(latitudeArc * 0.5);
latitudeH *= latitudeH;
double lontitudeH = sin(longitudeArc * 0.5);
lontitudeH *= lontitudeH;
double tmp = cos(from.lat*DEG_TO_RAD) * cos(to.lat*DEG_TO_RAD);
double arcInRadians = 2.0 * asin(sqrt(latitudeH + tmp*lontitudeH));
distance = EARTH_RADIUS_IN_METERS * arcInRadians;
return 0; //success
}

确实找到了解决我问题的东西,但还没有设法让它工作。不确定此链接中的描述是否遵循标准库。

我前段时间给出了一个类似问题的答案,并试图澄清语言环境的问题,所以如果你想了解更多:stod 在 boost::locale 中无法正常工作。简而言之:

  1. 在插件中调用std::setlocale并不是一件好事,因为它是整个应用程序的全局状态,您可能会弄乱应用程序的其他部分。至少您应该在完成后立即重置区域设置。

  2. 如果使用 std::stream ( stringstreamcincout (,则流的区域设置与创建流对象时全局C++区域设置的值匹配。通常它是经典的"C"区域设置。但是没有保证,因为应用程序的某些部分可以像您一样操纵它。

  3. 您可以通过调用 std::stream::imbue 来确保std::stream具有所需的区域设置。这只会操纵您的流,而不会操纵全局状态。

例如:

std::stringstream stream;
stream.imbue(std::locale::classic());//classical locale "C" with separator '.'