I'm working with a piece of .Net Core 6 library (dll), in which, amongst other functionality, numbers and date/time information is outputted as strings and this code is required to be globalized ie. to be culture-aware. The library is for a document producing system, so in the same application session, users would produce documents for de-CH, de-FR, en-US etc. So basically, We would like to have the library culture-aware, but the culture outputted should have it's "standard" formats, not the ones customised in Operating system level or user level.
For example for decimal values, the library code is using these methods to output stuff:
public string ToString();
public string ToString(string? format);
public string ToString(IFormatProvider? provider)
public string ToString(string? format, IFormatProvider? provider)
So calling these overloads by different users in different computers having different culture-settings, will format the output differently: decimal separator can be comma or point and the thousands grouping delimiter might be non-braking-space or ’ etc.
However, when provider
argument is passed explicitly by using an instance of a specific culture for example "fr-CH" (Swiss-French) and with CultureInfo.UseUserOverride == false, I was expecting always the same output string, independent of user and/or computer. So, I was thinking, that .Net would have "built-in" values for all the properties of CultureInfo-instances for all culture-types and with argument useUserOverride: false
I could force the formatting to use these "Built-in" values and hence would output always the same string.
But this seems not be the case!
The code:
CultureInfo cultInfo = new CultureInfo(name: "fr-CH", useUserOverride: false) //* fr-CH" is Swiss-French
Decimal myDec = -123456789123456.987654321M;
String output = myDec.ToString(format: "N4", provider: cultInfo); // I was expecting always same output independent, which user is calling it in which computer, but this is not the case!
This example code was compiled with .Net Core 6 as Console application and I run it on a) a Windows 10 Pro PC and b) on a Windows Server 2012R2 and they had having different outputs.
- Fr-CH/Windows Pro 10 PC: -123 456 789 123 456,9877
- Fr-CH/Windows Server 2012R2: -123'456'789'123'456.9877
The same by setting CultureInfo cultInfo = CultureInfo.InvariantCulture
: Then the result look always the same, which is expected.
I also tried setting Thread.CurrentThread.CurrentCulture
and Thread.CurrentThread.CurrentUICulture
using the cultInfo
-variable the above and then calling ToString(format?, provider?)
, but there was no effect: the results were exactly like without setting the CurrentThread
. PS. I actually learned from here, here and here that setting Thread.CurrentThread.CurrentUICulture
has nothing to do with formatting (my case), but with translation (resx stuff).
I'm aware of this vaste piece of Microsoft documentation about CultureInfo
-class, but I cannot see the tree from the forest there.
My questions are as follows:
- Has any version of .Net Core (or any version .Net Framework) a "built-in values" for all CultureInfo-properties at all?
- How could I force my code to use always exactly the culture given culture independent of the environment?
This would be very important in order to be able to unit-test the library output in a reliable way.
Likely important this would be, when reading back (utilizing System.Convert(value, provider?) the values outputted by any user in any computer (just knowing the culture they were outputted).
Set it on the thead for it to work and if you don't want it to repeat in every place:
when you run the code on other computers, try to check the cultureinfo and see what it will return. I bet it is not setting it to fr-CH: