How can I multiply a INumber<T> with an int?

2k Views Asked by At

How should I implement the multiplication by int (z * 2) on the last line?

public static TResult Test<TResult>() where TResult : INumber<TResult>
{
    TResult x = TResult.AdditiveIdentity;
    TResult y = TResult.MultiplicativeIdentity;
    TResult z = x * y;
    TResult z2 = z * 2; // <--- this gives the CS0019 error "The operator * cannot be applied to operands of type 'TResult' and 'int'
    return z2;
}

--- the suggested solution is to add an interface, but it breaks this: IMultiplyOperators<TResult, int, TResult>

public static void Tester()
{
    Test<decimal>(); // CS0315 Tye type decimal cannot be used as type parameter TResult..... there is no boxing conversion
}

For now I will inject the function myself and use:

public static TResult Test<TResult>(Func<TResult, int, TResult> mul) where TResult : INumber<TResult>
{
    TResult x = TResult.AdditiveIdentity;
    TResult y = TResult.MultiplicativeIdentity;
    TResult z = x * y;
    TResult z2 = mul(z, 2);
    return z2;
}
3

There are 3 best solutions below

3
On BEST ANSWER

I suggest converting 2 to TResult and only then multiply:

   TResult z2 = z * TResult.CreateChecked(2);

Here we create TResult instance from integer value 2, while checking 2 for overflow and similar possible errors (an exception will be thrown if 2 can't be converted into TResult: either OverflowException or NotSupportedException).

Code:

public static TResult Test<TResult>() where TResult : INumber<TResult>
{
    TResult x = TResult.AdditiveIdentity;
    TResult y = TResult.MultiplicativeIdentity;
    TResult z = x * y;
    TResult z2 = z * TResult.CreateChecked(2);
    return z2;
}

Demo:

Console.WriteLine(Test<decimal>());

Output:

0

Fiddle

2
On

While not a general solution, in this very case (multiplier = 2) you can simply write:

TResult z2 = z + z;

Except the z is TResult.AdditiveIdentity * TResult.MultiplicativeIdentity. But TResult.AdditiveIdentity is zero. So the result will be zero.

4
On

You need to add the IMultiplyOperators<, ,> interface:

public static TResult Test<TResult>() where TResult : INumber<TResult>, IMultiplyOperators<TResult, int, TResult>
{
    TResult x = TResult.AdditiveIdentity;
    TResult y = TResult.MultiplicativeIdentity;
    TResult z = x * y;
    TResult z2 = z * 2;
    return z2;
}

You can see the list of operator interfaces here.