If you look at the NUMBERFMT
structure, the way groups are expressed by the Grouping
member do not match the value returned by LOCALE_SGROUPING
:
LOCALE_ SGROUPING |
Grouping | Sample | Culture |
---|---|---|---|
3;0 | 3 | 1,234,567 | United States |
3;2;0 | 32 | 12,34,567 | India |
3 | 30 | 1234,567 | (none I know of) |
LOCALE_SGROUPING
expresses grouping as a series of semicolon-separated numbers, each expressing the number of digits in each group (least-significant group first). A trailing zero indicates that the last grouping should be repeated indefinitely. For example, “3;2;0” means “Group the three least significant digits, then in twos until you run out of digits.” If there is no trailing “;0”, then there are no commas past that point. For example, “3” means “Group the three least significant digits, then stop.”
The Grouping
member expresses the grouping rules differently. Each significant digit represents a group, with the most significant digit representing the least-significant group, with the units digit repeated indefinitely. For example, “32” means “make a group of three digits, then group by twos thereafter.” To suppress the repetition, multiply by ten.
In other words, the two systems are basically the same, with the Grouping
consisting of the LOCALE_SGROUPING
string with the semicolons removed. Except that the meaning of the trailing zero is reversed, so if LOCALE_SGROUPING
has a trailing zero, you have to remove it to get the Grouping
, and if it lacks a trailing zero, then you have to add one to the Grouping
.
It’s kind of strange that the two systems differ, considering that they both came from the same NLS team! It’s probably a case of parallel evolution, wherein the locale-string folks and the number-formatting folks came up with their respective systems independently.
Writing code to implement this conversion from LOCALE_SGROUPING
to Grouping
shouldn’t be hard once you understand the algorithm, so I’ll leave that as an exercise.
Fortunately, in real life you rarely have need to perform this conversion, for you can just pass the desired locale as the first parameter to the GetNumberFormat
(or even better, LOCALE_USER_DEFAULT
), pass a NULL
pointer as the lpNumberFormat
, and let NLS do all the work.
0 comments