Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How to change NaN string representation in C#?

My program saves a pointcloud to file, where each pointcloud is a Point3D[,], from the System.Windows.Media.Media3D namespace. This shows a line of the output file (in portuguese):

-112,644088741971;71,796623005014;NaN (Não é um número)

while I'd like it to be (on order to be correctly parsed afterwards):

-112,644088741971;71,796623005014;NaN

The block of code that generates the file is here:

var lines = new List<string>();

for (int rows = 0; rows < malha.GetLength(0); rows++) {
    for (int cols = 0; cols < malha.GetLength(1); cols++) {

        double x = coordenadas_x[cols];
        double y = coordenadas_y[rows];
        double z;

        if ( SomeTest() ) {
            z = alglib.rbfcalc2(model, x, y);
        } else {
            z = double.NaN;
        }

        var p = new Point3D(x, y, z);
        lines.Add(p.ToString());                       

        malha[rows, cols] = p;
    }
}

File.WriteAllLines("../../../../dummydata/malha.txt", lines);

It seems like the double.NaN.ToString() method, called from inside Point3D.ToString(), includes that parenthesized "additional explanation" which I don't want at all.

Is there a way to change/override this method so that it outputs only NaN, without the parentheses part?

like image 415
heltonbiker Avatar asked Mar 08 '13 18:03

heltonbiker


2 Answers

Double.ToString() uses NumberFormatInfo.CurrentInfo to format its numbers. This last property references to the CultureInfo that is currently set on the active thread. This defaults to the user's current locale. In this case its a Portuguese culture setting. To avoid this behavior, use the Double.ToString(IFormatProvider) overload. In this case you could use CultureInfo.InvariantCulture.

Additionally you can just switch the NaN symbol if you want to retain all other markup. By default globalization information is read only. Creating a clone will get around this.

System.Globalization.NumberFormatInfo numberFormatInfo = 
    (System.Globalization.NumberFormatInfo) System.Globalization.NumberFormatInfo.CurrentInfo.Clone();
numberFormatInfo.NaNSymbol = "NaN";

double num = double.NaN;
string numString = System.Number.FormatDouble(num, null, numberFormatInfo);

To set this on the current thread, create a copy of the current culture and set the number format info on the culture. Pre .NET 4.5 there's no way to set it for all threads. After creating each thread you would have to ensure a correct CultureInfo. As of .NET 4.5 there's CultureInfo.DefaultThreadCurrentCulture which defines the default culture for threads within the AppDomain. This setting is only considered when the culture of the thread has not been set yet (see MSDN).

Example for a single thread:

System.Globalization.CultureInfo myCulture =
     (System.Globalization.CultureInfo)System.Threading.Thread.CurrentThread.CurrentCulture.Clone();
myCulture.NumberFormat.NaNSymbol = "NaN";

System.Threading.Thread.CurrentThread.CurrentCulture = myCulture;   
string numString = double.NaN.ToString();
like image 124
Caramiriel Avatar answered Oct 10 '22 10:10

Caramiriel


Simply don't pass NaN values to ToString.

For example (wrapping in an extension method for easy reuse):

static string ToCleanString(this double val)
{
    if (double.IsNan(val)) return "NaN";
    return val.ToString();
}
like image 41
Ben Voigt Avatar answered Oct 10 '22 10:10

Ben Voigt