Runden auf Ganzzahlen in C# (C# >= 2.0)

Bearbeiten

Siehe auch Rundung und en:Rounding.

using System;
using System.Collections.Generic;

namespace DummyNamespace
{
	public static class Mathematics
	{
		/// <summary>
		/// Die Strategie, wie sich der Dezimalanteil einer Zahl auf das Rundungsergebnis auswirkt.
		/// Siehe auch http://en.wikipedia.org/wiki/Rounding.
		/// </summary>
		public enum RoundingPolicy
		{
			/// <summary>
			/// Das Ergebnis ist der Ganzzahlteil ohne Dezimalstellen ("Truncate").
			/// </summary>
			TowardsZero,

			/// <summary>
			/// Das Ergebnis ist die größte Ganzzahl, die nicht größer als der Eingabewert ist ("Floor").
			/// </summary>
			Down,

			/// <summary>
			/// Das Ergebnis ist die kleinste Ganzzahl, die nicht kleiner als der Eingabewert ist ("Ceiling").
			/// </summary>
			Up,

			/// <summary>
			/// Das Ergebnis ist der Eingabewert, falls der Eingabewert keine Dezimalstellen hat.
			/// Andernfalls ist das Ergebnis die Ganzzahl,
			/// die am nächsten zu Null liegt, wobei der Eingabewert zwischen Null und dem Ergebnis liegen muss.
			/// </summary>
			AwayFromZero,

			/// <summary>
			/// Das Ergebnis ist die Ganzzahl, die am nächsten zum Eingabewert liegt.
			/// Wenn der Dezimalteil des Eingabewertes gleich 1/2 ist,
			/// ist das Ergebnis der Eingabewert plus 1/2 für positive Eingabewerte,
			/// bzw. der Eingabewert minus 1/2 für negative Eingabewerte.
			/// </summary>
			ToNearest_HalfAwayFromZero,

			/// <summary>
			/// Das Ergebnis ist die Ganzzahl, die am nächsten zum Eingabewert liegt.
			/// Wenn der Dezimalteil des Eingabewertes gleich 1/2 ist,
			/// ist das Ergebnis die Gerade Zahl, die am nächsten zum Eingabewert liegt.
			/// </summary>
			ToNearest_HalfToEven,

			/// <summary>
			/// Das Ergebnis ist die Ganzzahl, die am nächsten zum Eingabewert liegt.
			/// Wenn der Dezimalteil des Eingabewertes gleich 1/2 ist,
			/// ist das Ergebnis der Eingabewert plus 1/2.
			/// </summary>
			ToNearest_HalfUp,
		}

		/// <summary>
		/// Ermittelt den ganzzahligen Teil einer Zahl für eine bestimmte Rundungs-Strategie (<see cref="RoundingPolicy" />).
		/// </summary>
		/// <param name="value">Die Eingabezahl.</param>
		/// <param name="roundingPolicy">Die Rundungs-Strategie.</param>
		/// <returns>Der ganzzahlige Teil der Eingabezahl gemäß der Rundungs-Strategie.</returns>
		public static Decimal GetRoundedInteger(Decimal value, RoundingPolicy roundingPolicy)
		{
			Decimal result;

			switch (roundingPolicy)
			{
				case RoundingPolicy.Down:
					{
						result = Math.Floor(value);
					}
					break;
				case RoundingPolicy.Up:
					{
						result = Math.Ceiling(value);
					}
					break;
				case RoundingPolicy.AwayFromZero:
					{
						result = (
							value >= 0 ?
							Math.Ceiling(value) :
							Math.Floor(value)
						);
					}
					break;
				case RoundingPolicy.TowardsZero:
					{
						result = Math.Truncate(value);
					}
					break;
				case RoundingPolicy.ToNearest_HalfAwayFromZero:
					{
						result = Math.Round(value, MidpointRounding.AwayFromZero);
					}
					break;
				case RoundingPolicy.ToNearest_HalfToEven:
					{
						result = Math.Round(value, MidpointRounding.ToEven);
					}
					break;
				case RoundingPolicy.ToNearest_HalfUp:
					{
						result = (
							IsHalf(value) ?
							Math.Ceiling(value) :
							Math.Round(value)
						);
					}
					break;
				default:
					throw new NotImplementedException(roundingPolicy.ToString());
			}

			return result;
		}

		/// <summary>
		/// Ermittelt, ob der vorzeichenlose Dezimalteil einer Zahl 1/2 ist.
		/// </summary>
		/// <param name="value">Die Eingabezahl.</param>
		/// <returns>True, wenn der vorzeichenlose Dezimalteil einer Zahl 1/2 ist. Sonst false.</returns>
		public static bool IsHalf(Decimal value)
		{
			Decimal fraction = Fraction(value);
			bool result = (fraction == new Decimal(0.5));

			return result;
		}

		/// <summary>
		/// Ermittelt den vorzeichenlosen Dezimalteil einer Zahl.
		/// </summary>
		/// <param name="value">Die Eingabezahl.</param>
		/// <returns>Der vorzeichenlose Dezimalteil.</returns>
		public static Decimal Fraction(Decimal value)
		{
			Decimal result = Math.Abs(value - Math.Truncate(value));

			return result;
		}
	}

	public class RoundedIntegerTester
	{
		static void Main(string[] args)
		{
			List<Decimal> values = new List<decimal>();
			{
				//Die 0 einfügen
				values.Add(0m);

				//Beträge der zu testenden Zahlen (außer 0)
				{
					//definieren...
					List<Decimal> absoluteValues = new List<Decimal>(
						new Decimal[] {
							    0.01m, 0.5m, 0.99m, 
							1m, 1.01m, 1.5m, 1.99m, 
							2m, 2.01m, 2.5m, 2.99m, 
							3m, 3.01m, 3.5m, 3.99m, 
							4m, 4.01m
						}
					);

					//... und einfügen
					foreach (Decimal absoluteValue in absoluteValues)
					{
						values.Add(+absoluteValue);
						values.Add(-absoluteValue);
					}
				}
			}
			values.Sort(); //zu testende Zahlen sortieren

			//alle möglichen RoundingPolicies holen
			Array roundingPolicies = Enum.GetValues(typeof(Mathematics.RoundingPolicy));

			//LOG-Datei erstellen
			using (System.IO.StreamWriter writer = new System.IO.StreamWriter(@"c:\RoundedIntegerTester.log"))
			{
				//Trennzeichen
				string separator = ";";

				//Kopfzeile
				{
					string line = "x" + separator;

					foreach (Mathematics.RoundingPolicy roundingPolicy in roundingPolicies)
					{
						line += roundingPolicy + separator;
					}

					writer.WriteLine(line);
				}

				//Zahlen testen
				{
					foreach (Decimal value in values)
					{
						string line = value + separator;

						foreach (Mathematics.RoundingPolicy roundingPolicy in roundingPolicies)
						{
							//Ergebnis holen
							Decimal y = Mathematics.GetRoundedInteger(value, roundingPolicy);

							line += y + separator;
						}

						writer.WriteLine(line);
					}
				}
			}

			//Datei für MediaWiki-Tabelle erstellen
			using (System.IO.StreamWriter writer = new System.IO.StreamWriter(@"c:\RoundedIntegerTester.MediaWikiTable"))
			{
				//Tabelle
				writer.WriteLine(
					"{|" +
					" class=" + "\"" + "wikitable" + "\""
				);
				{
					//Kopfzeile
					{
						writer.WriteLine("! " + "x");

						foreach (Mathematics.RoundingPolicy roundingPolicy in roundingPolicies)
						{
							writer.WriteLine("! " + roundingPolicy);
						}
					}

					//Zahlen testen
					{
						foreach (Decimal value in values)
						{
							writer.WriteLine("|" + "-");
							writer.WriteLine("| " + value);

							foreach (Mathematics.RoundingPolicy roundingPolicy in roundingPolicies)
							{
								//Ergebnis holen
								Decimal y = Mathematics.GetRoundedInteger(value, roundingPolicy);

								writer.WriteLine("| " + y);
							}
						}
					}
				}

				writer.WriteLine("|}");
			}
		}
	}
}


Das Ergebnis der Ganzzahlrundung sollte folgende Werte haben:

x Down Up AwayFromZero TowardsZero ToNearest_HalfAwayFromZero ToNearest_HalfToEven ToNearest_HalfUp
-4,01 -5 -4 -5 -4 -4 -4 -4
-4 -4 -4 -4 -4 -4 -4 -4
-3,99 -4 -3 -4 -3 -4 -4 -4
-3,5 -4 -3 -4 -3 -4 -4 -3
-3,01 -4 -3 -4 -3 -3 -3 -3
-3 -3 -3 -3 -3 -3 -3 -3
-2,99 -3 -2 -3 -2 -3 -3 -3
-2,5 -3 -2 -3 -2 -3 -2 -2
-2,01 -3 -2 -3 -2 -2 -2 -2
-2 -2 -2 -2 -2 -2 -2 -2
-1,99 -2 -1 -2 -1 -2 -2 -2
-1,5 -2 -1 -2 -1 -2 -2 -1
-1,01 -2 -1 -2 -1 -1 -1 -1
-1 -1 -1 -1 -1 -1 -1 -1
-0,99 -1 0 -1 0 -1 -1 -1
-0,5 -1 0 -1 0 -1 0 0
-0,01 -1 0 -1 0 0 0 0
0 0 0 0 0 0 0 0
0,01 0 1 1 0 0 0 0
0,5 0 1 1 0 1 0 1
0,99 0 1 1 0 1 1 1
1 1 1 1 1 1 1 1
1,01 1 2 2 1 1 1 1
1,5 1 2 2 1 2 2 2
1,99 1 2 2 1 2 2 2
2 2 2 2 2 2 2 2
2,01 2 3 3 2 2 2 2
2,5 2 3 3 2 3 2 3
2,99 2 3 3 2 3 3 3
3 3 3 3 3 3 3 3
3,01 3 4 4 3 3 3 3
3,5 3 4 4 3 4 4 4
3,99 3 4 4 3 4 4 4
4 4 4 4 4 4 4 4
4,01 4 5 5 4 4 4 4