首页 > 编程 > .NET > 正文

C标准库源码解剖(10):区域设置函数locale.h

2024-07-10 13:27:17
字体:
来源:转载
供稿:网友

国家、文化和语言规则集称为区域设置,locale.h头文件中定义了区域设置相关的函数。setlocale函数用于设置或返回当前的区域特性,localeconv用于返回当前区域中的数字和货币信息(保存在struct lconv结构实例中)。setlocale的第一个实参指定要改变的区域行为类别,预定义的setlocale类别有:
    LC_ALL:所有行为。
    LC_COLLATE:strcoll和strxfrm函数的行为。
    LC_CTYPE:ctype.h中的字符串处理函数行为。
    LC_MONETARY:localeconv返回的币值信息。
    LC_NUMERIC:localeconv返回的小数点和非币值信息。
    LC_TIME:strftime函数的行为。
    locale.h中内容如下:

 

[cpp] view plaincopy
  1. /* ISO C99 Standard: 7.11 本地化   <locale.h>  */  
  2. #ifndef _LOCALE_H  
  3. #define _LOCALE_H   1  
  4. #include <features.h>  
  5. #define __need_NULL  
  6. #include <stddef.h>  
  7. #include <bits/locale.h>  
  8. __BEGIN_DECLS  
  9. /* 下面是setlocale第一个实参可能的值。代码假设LC_*宏的最小值为0 */  
  10. #define LC_CTYPE          __LC_CTYPE  
  11. #define LC_NUMERIC        __LC_NUMERIC  
  12. #define LC_TIME           __LC_TIME  
  13. #define LC_COLLATE        __LC_COLLATE  
  14. #define LC_MONETARY       __LC_MONETARY  
  15. #define LC_MESSAGES       __LC_MESSAGES  
  16. #define LC_ALL        __LC_ALL  
  17. #define LC_PAPER      __LC_PAPER  
  18. #define LC_NAME       __LC_NAME  
  19. #define LC_ADDRESS    __LC_ADDRESS  
  20. #define LC_TELEPHONE      __LC_TELEPHONE  
  21. #define LC_MEASUREMENT    __LC_MEASUREMENT  
  22. #define LC_IDENTIFICATION __LC_IDENTIFICATION  
  23.   
  24. __BEGIN_NAMESPACE_STD  
  25. /* 获取数字和货币记号的结构  */  
  26. struct lconv  
  27. {  
  28.   /* 数字(非币值)信息  */  
  29.   char *decimal_point;      /* 数字的小数点号  */  
  30.   char *thousands_sep;      /* 数字的千分组分隔符  */  
  31.   /* 每个元素为相应组中的数字位数,索引越高的元素越靠左边。一个值为CHAR_MAX的元素表示没有更多 
  32.       的分组了。一个值为0的元素表示前面的元素能用在靠左边的所有分组中 */  
  33.   char *grouping;  /* 数字分组分隔符 */  
  34.   /* 货币信息  */  
  35.   /* 前面的三个字符ISO 4217中规定的货币符号,第四个字符是分隔符,第五个字符是'/0' */  
  36.   char *int_curr_symbol;  
  37.   char *currency_symbol;    /* 本地货币符号  */  
  38.   char *mon_decimal_point;  /* 货币的小数点号  */  
  39.   char *mon_thousands_sep;  /* 倾向的千分组分隔符  */  
  40.   char *mon_grouping;       /* 类似于grouping元素(参考上面)  */  
  41.   char *positive_sign;      /* 正币值的符号  */  
  42.   char *negative_sign;      /* 负币值的符号  */  
  43.   char int_frac_digits;     /* 国际币值的小数部分  */  
  44.   char frac_digits;     /* 本地币值的小数部分  */  
  45.   /* 如果currency_symbol放在正币值之前则为1,否则为0  */  
  46.   char p_cs_precedes;  
  47.   /* 当且仅当currency_symbol与正币值之间用空格分开时为1  */  
  48.   char p_sep_by_space;  
  49.   /* 如果currency_symbol放在负币值之前则为1,否则为0  */  
  50.   char n_cs_precedes;  
  51.   /* 当且仅当currency_symbol与负币值之间用空格分开时为1  */  
  52.   char n_sep_by_space;  
  53.   /* 正值和负值符号的定位: 
  54.      0 用圆括号括住数量和currency_symbol 
  55.      1 正负号放在数量和currency_symbol之前 
  56.      2 正负号放在数量和currency_symbol之后 
  57.      3 正负号正好放在currency_symbol之前 
  58.      4 正负号正好放在currency_symbol之后  */  
  59.   char p_sign_posn;  
  60.   char n_sign_posn;  
  61. #ifdef __USE_ISOC99  
  62.   /* 如果int_curr_symbol放在正币值之前则为1,否则为0  */  
  63.   char int_p_cs_precedes;  
  64.   /* 当且仅当int_curr_symbol与正币值之间用空格分开时为1  */  
  65.   char int_p_sep_by_space;  
  66.   /* 如果int_curr_symbol放在负币值之前则为1,否则为0   */  
  67.   char int_n_cs_precedes;  
  68.   /* 当且仅当int_curr_symbol与负币值之间用空格分开时为1  */  
  69.   char int_n_sep_by_space;  
  70.   /* 正值和负值符号的定位: 
  71.      0 用圆括号括住数量和int_curr_symbol 
  72.      1 正负号放在数量和int_curr_symbol之前 
  73.      2 正负号放在数量和int_curr_symbol之后 
  74.      3 正负号正好放在int_curr_symbol之前 
  75.      4 正负号正好放在int_curr_symbol之后  */  
  76.   char int_p_sign_posn;  
  77.   char int_n_sign_posn;  
  78. #else  
  79.   char __int_p_cs_precedes;  
  80.   char __int_p_sep_by_space;  
  81.   char __int_n_cs_precedes;  
  82.   char __int_n_sep_by_space;  
  83.   char __int_p_sign_posn;  
  84.   char __int_n_sign_posn;  
  85. #endif  
  86. };  
  87.   
  88. /* 设置或返回当前的区域特性  */  
  89. extern char *setlocale (int __category, __const char *__locale) __THROW;  
  90. /* 返回当前区域中的数字和货币信息  */  
  91. extern struct lconv *localeconv (void) __THROW;  
  92. __END_NAMESPACE_STD  
  93.   
  94. /* 下面是一些扩展 */  
  95. __END_DECLS  
  96. #endif /* locale.h  */  

 

    setlocale函数的实现代码比较复杂,这里就不解剖了。localeconv函数的代码比较简单,它先创建一个struct lconv实例result用来保存返回的信息,然后用编译器的内部宏来获取数字和货币的各个成员值,存入result中,然后返回。如下:

 

[cpp] view plaincopy
  1. /* localeconv.c:localeconv函数的实现  */  
  2. #include <locale.h>  
  3. #include "localeinfo.h"  
  4. #include <shlib-compat.h>  
  5. /* 返回当前区域中的数字和货币信息  */  
  6. struct lconv *  
  7. __localeconv (void)  
  8. {  
  9.   static struct lconv result; /* 用于保存返回值的lconv类型实例 */  
  10.   result.decimal_point = (char *) _NL_CURRENT (LC_NUMERIC, DECIMAL_POINT);  
  11.   result.thousands_sep = (char *) _NL_CURRENT (LC_NUMERIC, THOUSANDS_SEP);  
  12.   result.grouping = (char *) _NL_CURRENT (LC_NUMERIC, GROUPING);  
  13.   if (*result.grouping == CHAR_MAX || *result.grouping == (char) -1)  
  14.     result.grouping = (char *) "";  
  15.   result.int_curr_symbol = (char *) _NL_CURRENT (LC_MONETARY, INT_CURR_SYMBOL);  
  16.   result.currency_symbol = (char *) _NL_CURRENT (LC_MONETARY, CURRENCY_SYMBOL);  
  17.   result.mon_decimal_point = (char *) _NL_CURRENT (LC_MONETARY,  
  18.                            MON_DECIMAL_POINT);  
  19.   result.mon_thousands_sep = (char *) _NL_CURRENT (LC_MONETARY,  
  20.                            MON_THOUSANDS_SEP);  
  21.   result.mon_grouping = (char *) _NL_CURRENT (LC_MONETARY, MON_GROUPING);  
  22.   if (*result.mon_grouping == CHAR_MAX || *result.mon_grouping == (char) -1)  
  23.     result.mon_grouping = (char *) "";  
  24.   result.positive_sign = (char *) _NL_CURRENT (LC_MONETARY, POSITIVE_SIGN);  
  25.   result.negative_sign = (char *) _NL_CURRENT (LC_MONETARY, NEGATIVE_SIGN);  
  26.   result.int_frac_digits = *(char *) _NL_CURRENT (LC_MONETARY,  
  27.                           INT_FRAC_DIGITS);  
  28.   result.frac_digits = *(char *) _NL_CURRENT (LC_MONETARY, FRAC_DIGITS);  
  29.   result.p_cs_precedes = *(char *) _NL_CURRENT (LC_MONETARY, P_CS_PRECEDES);  
  30.   result.p_sep_by_space = *(char *) _NL_CURRENT (LC_MONETARY, P_SEP_BY_SPACE);  
  31.   result.n_cs_precedes = *(char *) _NL_CURRENT (LC_MONETARY, N_CS_PRECEDES);  
  32.   result.n_sep_by_space = *(char *) _NL_CURRENT (LC_MONETARY, N_SEP_BY_SPACE);  
  33.   result.p_sign_posn = *(char *) _NL_CURRENT (LC_MONETARY, P_SIGN_POSN);  
  34.   result.n_sign_posn = *(char *) _NL_CURRENT (LC_MONETARY, N_SIGN_POSN);  
  35.   result.int_p_cs_precedes = *(char *) _NL_CURRENT (LC_MONETARY,  
  36.                             INT_P_CS_PRECEDES);  
  37.   result.int_p_sep_by_space = *(char *) _NL_CURRENT (LC_MONETARY,  
  38.                              INT_P_SEP_BY_SPACE);  
  39.   result.int_n_cs_precedes = *(char *) _NL_CURRENT (LC_MONETARY,  
  40.                             INT_N_CS_PRECEDES);  
  41.   result.int_n_sep_by_space = *(char *) _NL_CURRENT (LC_MONETARY,  
  42.                              INT_N_SEP_BY_SPACE);  
  43.   result.int_p_sign_posn = *(char *) _NL_CURRENT (LC_MONETARY,  
  44.                           INT_P_SIGN_POSN);  
  45.   result.int_n_sign_posn = *(char *) _NL_CURRENT (LC_MONETARY,  
  46.                           INT_N_SIGN_POSN);  
  47.   return &result;  
  48. }  
  49. versioned_symbol (libc, __localeconv, localeconv, GLIBC_2_2);  
  50. #if SHLIB_COMPAT (libc, GLIBC_2_0, GLIBC_2_2)  
  51. strong_alias (__localeconv, __localeconv20)  
  52. compat_symbol (libc, __localeconv20, localeconv, GLIBC_2_0);  
  53. #endif  

 

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表