如何使gzprintf
(或一般来说是printf)使用“。”作为小数点分隔符,可以有效地忽略LC_NUMERIC或使用“ LC_NUMERIC = C”。
背景:我的程序使用gettext
进行setlocale (LC_ALL, ....)
来更改其语言环境。但有一点我需要写入一个测量文件,该文件应具有固定的“”。作为小数点分隔符。我当然可以做类似的事情
const char *old = setlocale (LC_NUMERIC, NULL);
... make copy
setlocale (LC_NUMERIC, "C");
... use gzprintf
setlocale (LC_NUMERIC, old_copy);
但是我担心运行时出现问题,因为我必须向文件中写入2kHz的样本,但尚未对其进行基准测试。
答案 0 :(得分:2)
不仅每次通话都用setlocale
包装,这样可能会很慢。这是完全线程不安全的,因此在一般/库代码中使用是不安全的。
假设您使用的是POSIX或类似POSIX的系统,那么您需要的是newlocale
/ uselocale
。一次执行以下操作(可能会很慢)并保存结果:
locale_t safe_locale = newlocale(LC_NUMERIC_MASK, "C", duplocale(LC_GLOBAL_LOCALE));
然后,每当您要使用它时:
locale_t old = uselocale(safe_locale);
/* Do stuff with it. */
uselocale(old);
您不必在每次操作之前来回切换,仅在返回可能期望其语言环境不受干扰的调用者之前,但是无论如何像这样来回切换都不会太慢。
或者,根本不尊重您的应用程序中的LC_NUMERIC
。调用setlocale(LC_ALL, "")
根据用户设置设置语言环境后,请调用setlocale(LC_NUMERIC, "C")
并保持原来的状态。这在库代码中是无效/不可接受的,但是如果您正在编写库代码,则可以平移并让调用者负责执行它认为合适的任何方法(绝对不要兑现LC_NUMERIC
或使用{{ 1}} / newlocale
进行切换)。您既可以在合同的文档中写下此要求,也可以将未履行合同的情况视为未定义,或者可以通过断言uselocale
指向非'.'
基数来积极测试字符串localeconv()->decimal_point
(并返回错误或如果失败则中止)。