Brak operatora wykładniczego dla języka C # był dla nas dużą uciążliwością, gdy szukaliśmy nowego języka do konwersji naszego oprogramowania obliczeniowego z dobrej starej wersji vb6.
Cieszę się, że wybraliśmy C #, ale wciąż denerwuje mnie, gdy piszę złożone równanie, w tym wykładniki. Metoda Math.Pow () sprawia, że równania są dość trudne do odczytania przez IMO.
Naszym rozwiązaniem było stworzenie specjalnej klasy DoubleX, w której przesłonimy operator ^ (patrz poniżej)
Działa to dość dobrze, o ile deklarujesz przynajmniej jedną ze zmiennych jako DoubleX:
DoubleX a = 2;
DoubleX b = 3;
Console.WriteLine($"a = {a}, b = {b}, a^b = {a ^ b}");
lub użyj jawnego konwertera na standardowe podwójne:
double c = 2;
double d = 3;
Console.WriteLine($"c = {c}, d = {d}, c^d = {c ^ (DoubleX)d}"); // Need explicit converter
Jednym z problemów związanych z tą metodą jest to, że wykładnik obliczany jest w niewłaściwej kolejności w porównaniu do innych operatorów. Można tego uniknąć, zawsze umieszczając dodatkową () wokół operacji, co ponownie utrudnia odczytanie równań:
DoubleX a = 2;
DoubleX b = 3;
Console.WriteLine($"a = {a}, b = {b}, 3+a^b = {3 + a ^ b}"); // Wrong result
Console.WriteLine($"a = {a}, b = {b}, 3+a^b = {3 + (a ^ b)}"); // Correct result
Mam nadzieję, że może to pomóc innym, którzy używają wielu złożonych równań w swoim kodzie, a może ktoś nawet ma pomysł, jak ulepszyć tę metodę ?! :-)
Klasa DoubleX:
using System;
namespace ExponentialOperator
{
/// <summary>
/// Double class that uses ^ as exponential operator
/// </summary>
public class DoubleX
{
#region ---------------- Fields ----------------
private readonly double _value;
#endregion ------------- Fields ----------------
#region -------------- Properties --------------
public double Value
{
get { return _value; }
}
#endregion ----------- Properties --------------
#region ------------- Constructors -------------
public DoubleX(double value)
{
_value = value;
}
public DoubleX(int value)
{
_value = Convert.ToDouble(value);
}
#endregion ---------- Constructors -------------
#region --------------- Methods ----------------
public override string ToString()
{
return _value.ToString();
}
#endregion ------------ Methods ----------------
#region -------------- Operators ---------------
// Change the ^ operator to be used for exponents.
public static DoubleX operator ^(DoubleX value, DoubleX exponent)
{
return Math.Pow(value, exponent);
}
public static DoubleX operator ^(DoubleX value, double exponent)
{
return Math.Pow(value, exponent);
}
public static DoubleX operator ^(double value, DoubleX exponent)
{
return Math.Pow(value, exponent);
}
public static DoubleX operator ^(DoubleX value, int exponent)
{
return Math.Pow(value, exponent);
}
#endregion ----------- Operators ---------------
#region -------------- Converters --------------
// Allow implicit convertion
public static implicit operator DoubleX(double value)
{
return new DoubleX(value);
}
public static implicit operator DoubleX(int value)
{
return new DoubleX(value);
}
public static implicit operator Double(DoubleX value)
{
return value._value;
}
#endregion ----------- Converters --------------
}
}
**
jako operatora potęgowania potęgi.