Is there a constraint that restricts my generic method to numeric types?

后端 未结 21 2682
栀梦
栀梦 2020-11-21 05:48

Can anyone tell me if there is a way with generics to limit a generic type argument T to only:

  • Int16
  • Int32
21条回答
  •  眼角桃花
    2020-11-21 06:11

    If you are using .NET 4.0 and later then you can just use dynamic as method argument and check in runtime that the passed dynamic argument type is numeric/integer type.

    If the type of the passed dynamic is not numeric/integer type then throw exception.

    An example short code that implements the idea is something like:

    using System;
    public class InvalidArgumentException : Exception
    {
        public InvalidArgumentException(string message) : base(message) {}
    }
    public class InvalidArgumentTypeException : InvalidArgumentException
    {
        public InvalidArgumentTypeException(string message) : base(message) {}
    }
    public class ArgumentTypeNotIntegerException : InvalidArgumentTypeException
    {
        public ArgumentTypeNotIntegerException(string message) : base(message) {}
    }
    public static class Program
    {
        private static bool IntegerFunction(dynamic n)
        {
            if (n.GetType() != typeof(Int16) &&
                n.GetType() != typeof(Int32) &&
                n.GetType() != typeof(Int64) &&
                n.GetType() != typeof(UInt16) &&
                n.GetType() != typeof(UInt32) &&
                n.GetType() != typeof(UInt64))
                throw new ArgumentTypeNotIntegerException("argument type is not integer type");
            //code that implements IntegerFunction goes here
        }
        private static void Main()
        {
             Console.WriteLine("{0}",IntegerFunction(0)); //Compiles, no run time error and first line of output buffer is either "True" or "False" depends on the code that implements "Program.IntegerFunction" static method.
             Console.WriteLine("{0}",IntegerFunction("string")); //Also compiles but it is run time error and exception of type "ArgumentTypeNotIntegerException" is thrown here.
             Console.WriteLine("This is the last Console.WriteLine output"); //Never reached and executed due the run time error and the exception thrown on the second line of Program.Main static method.
        }
    

    Of course that this solution works in run time only but never in compile time.

    If you want a solution that always works in compile time and never in run time then you will have to wrap the dynamic with a public struct/class whose overloaded public constructors accept arguments of the desired types only and give the struct/class appropriate name.

    It makes sense that the wrapped dynamic is always private member of the class/struct and it is the only member of the struct/class and the name of the only member of the struct/class is "value".

    You will also have to define and implement public methods and/or operators that work with the desired types for the private dynamic member of the class/struct if necessary.

    It also makes sense that the struct/class has special/unique constructor that accepts dynamic as argument that initializes it's only private dynamic member called "value" but the modifier of this constructor is private of course.

    Once the class/struct is ready define the argument's type of IntegerFunction to be that class/struct that has been defined.

    An example long code that implements the idea is something like:

    using System;
    public struct Integer
    {
        private dynamic value;
        private Integer(dynamic n) { this.value = n; }
        public Integer(Int16 n) { this.value = n; }
        public Integer(Int32 n) { this.value = n; }
        public Integer(Int64 n) { this.value = n; }
        public Integer(UInt16 n) { this.value = n; }
        public Integer(UInt32 n) { this.value = n; }
        public Integer(UInt64 n) { this.value = n; }
        public Integer(Integer n) { this.value = n.value; }
        public static implicit operator Int16(Integer n) { return n.value; }
        public static implicit operator Int32(Integer n) { return n.value; }
        public static implicit operator Int64(Integer n) { return n.value; }
        public static implicit operator UInt16(Integer n) { return n.value; }
        public static implicit operator UInt32(Integer n) { return n.value; }
        public static implicit operator UInt64(Integer n) { return n.value; }
        public static Integer operator +(Integer x, Int16 y) { return new Integer(x.value + y); }
        public static Integer operator +(Integer x, Int32 y) { return new Integer(x.value + y); }
        public static Integer operator +(Integer x, Int64 y) { return new Integer(x.value + y); }
        public static Integer operator +(Integer x, UInt16 y) { return new Integer(x.value + y); }
        public static Integer operator +(Integer x, UInt32 y) { return new Integer(x.value + y); }
        public static Integer operator +(Integer x, UInt64 y) { return new Integer(x.value + y); }
        public static Integer operator -(Integer x, Int16 y) { return new Integer(x.value - y); }
        public static Integer operator -(Integer x, Int32 y) { return new Integer(x.value - y); }
        public static Integer operator -(Integer x, Int64 y) { return new Integer(x.value - y); }
        public static Integer operator -(Integer x, UInt16 y) { return new Integer(x.value - y); }
        public static Integer operator -(Integer x, UInt32 y) { return new Integer(x.value - y); }
        public static Integer operator -(Integer x, UInt64 y) { return new Integer(x.value - y); }
        public static Integer operator *(Integer x, Int16 y) { return new Integer(x.value * y); }
        public static Integer operator *(Integer x, Int32 y) { return new Integer(x.value * y); }
        public static Integer operator *(Integer x, Int64 y) { return new Integer(x.value * y); }
        public static Integer operator *(Integer x, UInt16 y) { return new Integer(x.value * y); }
        public static Integer operator *(Integer x, UInt32 y) { return new Integer(x.value * y); }
        public static Integer operator *(Integer x, UInt64 y) { return new Integer(x.value * y); }
        public static Integer operator /(Integer x, Int16 y) { return new Integer(x.value / y); }
        public static Integer operator /(Integer x, Int32 y) { return new Integer(x.value / y); }
        public static Integer operator /(Integer x, Int64 y) { return new Integer(x.value / y); }
        public static Integer operator /(Integer x, UInt16 y) { return new Integer(x.value / y); }
        public static Integer operator /(Integer x, UInt32 y) { return new Integer(x.value / y); }
        public static Integer operator /(Integer x, UInt64 y) { return new Integer(x.value / y); }
        public static Integer operator %(Integer x, Int16 y) { return new Integer(x.value % y); }
        public static Integer operator %(Integer x, Int32 y) { return new Integer(x.value % y); }
        public static Integer operator %(Integer x, Int64 y) { return new Integer(x.value % y); }
        public static Integer operator %(Integer x, UInt16 y) { return new Integer(x.value % y); }
        public static Integer operator %(Integer x, UInt32 y) { return new Integer(x.value % y); }
        public static Integer operator %(Integer x, UInt64 y) { return new Integer(x.value % y); }
        public static Integer operator +(Integer x, Integer y) { return new Integer(x.value + y.value); }
        public static Integer operator -(Integer x, Integer y) { return new Integer(x.value - y.value); }
        public static Integer operator *(Integer x, Integer y) { return new Integer(x.value * y.value); }
        public static Integer operator /(Integer x, Integer y) { return new Integer(x.value / y.value); }
        public static Integer operator %(Integer x, Integer y) { return new Integer(x.value % y.value); }
        public static bool operator ==(Integer x, Int16 y) { return x.value == y; }
        public static bool operator !=(Integer x, Int16 y) { return x.value != y; }
        public static bool operator ==(Integer x, Int32 y) { return x.value == y; }
        public static bool operator !=(Integer x, Int32 y) { return x.value != y; }
        public static bool operator ==(Integer x, Int64 y) { return x.value == y; }
        public static bool operator !=(Integer x, Int64 y) { return x.value != y; }
        public static bool operator ==(Integer x, UInt16 y) { return x.value == y; }
        public static bool operator !=(Integer x, UInt16 y) { return x.value != y; }
        public static bool operator ==(Integer x, UInt32 y) { return x.value == y; }
        public static bool operator !=(Integer x, UInt32 y) { return x.value != y; }
        public static bool operator ==(Integer x, UInt64 y) { return x.value == y; }
        public static bool operator !=(Integer x, UInt64 y) { return x.value != y; }
        public static bool operator ==(Integer x, Integer y) { return x.value == y.value; }
        public static bool operator !=(Integer x, Integer y) { return x.value != y.value; }
        public override bool Equals(object obj) { return this == (Integer)obj; }
        public override int GetHashCode() { return this.value.GetHashCode(); }
        public override string ToString() { return this.value.ToString(); }
        public static bool operator >(Integer x, Int16 y) { return x.value > y; }
        public static bool operator <(Integer x, Int16 y) { return x.value < y; }
        public static bool operator >(Integer x, Int32 y) { return x.value > y; }
        public static bool operator <(Integer x, Int32 y) { return x.value < y; }
        public static bool operator >(Integer x, Int64 y) { return x.value > y; }
        public static bool operator <(Integer x, Int64 y) { return x.value < y; }
        public static bool operator >(Integer x, UInt16 y) { return x.value > y; }
        public static bool operator <(Integer x, UInt16 y) { return x.value < y; }
        public static bool operator >(Integer x, UInt32 y) { return x.value > y; }
        public static bool operator <(Integer x, UInt32 y) { return x.value < y; }
        public static bool operator >(Integer x, UInt64 y) { return x.value > y; }
        public static bool operator <(Integer x, UInt64 y) { return x.value < y; }
        public static bool operator >(Integer x, Integer y) { return x.value > y.value; }
        public static bool operator <(Integer x, Integer y) { return x.value < y.value; }
        public static bool operator >=(Integer x, Int16 y) { return x.value >= y; }
        public static bool operator <=(Integer x, Int16 y) { return x.value <= y; }
        public static bool operator >=(Integer x, Int32 y) { return x.value >= y; }
        public static bool operator <=(Integer x, Int32 y) { return x.value <= y; }
        public static bool operator >=(Integer x, Int64 y) { return x.value >= y; }
        public static bool operator <=(Integer x, Int64 y) { return x.value <= y; }
        public static bool operator >=(Integer x, UInt16 y) { return x.value >= y; }
        public static bool operator <=(Integer x, UInt16 y) { return x.value <= y; }
        public static bool operator >=(Integer x, UInt32 y) { return x.value >= y; }
        public static bool operator <=(Integer x, UInt32 y) { return x.value <= y; }
        public static bool operator >=(Integer x, UInt64 y) { return x.value >= y; }
        public static bool operator <=(Integer x, UInt64 y) { return x.value <= y; }
        public static bool operator >=(Integer x, Integer y) { return x.value >= y.value; }
        public static bool operator <=(Integer x, Integer y) { return x.value <= y.value; }
        public static Integer operator +(Int16 x, Integer y) { return new Integer(x + y.value); }
        public static Integer operator +(Int32 x, Integer y) { return new Integer(x + y.value); }
        public static Integer operator +(Int64 x, Integer y) { return new Integer(x + y.value); }
        public static Integer operator +(UInt16 x, Integer y) { return new Integer(x + y.value); }
        public static Integer operator +(UInt32 x, Integer y) { return new Integer(x + y.value); }
        public static Integer operator +(UInt64 x, Integer y) { return new Integer(x + y.value); }
        public static Integer operator -(Int16 x, Integer y) { return new Integer(x - y.value); }
        public static Integer operator -(Int32 x, Integer y) { return new Integer(x - y.value); }
        public static Integer operator -(Int64 x, Integer y) { return new Integer(x - y.value); }
        public static Integer operator -(UInt16 x, Integer y) { return new Integer(x - y.value); }
        public static Integer operator -(UInt32 x, Integer y) { return new Integer(x - y.value); }
        public static Integer operator -(UInt64 x, Integer y) { return new Integer(x - y.value); }
        public static Integer operator *(Int16 x, Integer y) { return new Integer(x * y.value); }
        public static Integer operator *(Int32 x, Integer y) { return new Integer(x * y.value); }
        public static Integer operator *(Int64 x, Integer y) { return new Integer(x * y.value); }
        public static Integer operator *(UInt16 x, Integer y) { return new Integer(x * y.value); }
        public static Integer operator *(UInt32 x, Integer y) { return new Integer(x * y.value); }
        public static Integer operator *(UInt64 x, Integer y) { return new Integer(x * y.value); }
        public static Integer operator /(Int16 x, Integer y) { return new Integer(x / y.value); }
        public static Integer operator /(Int32 x, Integer y) { return new Integer(x / y.value); }
        public static Integer operator /(Int64 x, Integer y) { return new Integer(x / y.value); }
        public static Integer operator /(UInt16 x, Integer y) { return new Integer(x / y.value); }
        public static Integer operator /(UInt32 x, Integer y) { return new Integer(x / y.value); }
        public static Integer operator /(UInt64 x, Integer y) { return new Integer(x / y.value); }
        public static Integer operator %(Int16 x, Integer y) { return new Integer(x % y.value); }
        public static Integer operator %(Int32 x, Integer y) { return new Integer(x % y.value); }
        public static Integer operator %(Int64 x, Integer y) { return new Integer(x % y.value); }
        public static Integer operator %(UInt16 x, Integer y) { return new Integer(x % y.value); }
        public static Integer operator %(UInt32 x, Integer y) { return new Integer(x % y.value); }
        public static Integer operator %(UInt64 x, Integer y) { return new Integer(x % y.value); }
        public static bool operator ==(Int16 x, Integer y) { return x == y.value; }
        public static bool operator !=(Int16 x, Integer y) { return x != y.value; }
        public static bool operator ==(Int32 x, Integer y) { return x == y.value; }
        public static bool operator !=(Int32 x, Integer y) { return x != y.value; }
        public static bool operator ==(Int64 x, Integer y) { return x == y.value; }
        public static bool operator !=(Int64 x, Integer y) { return x != y.value; }
        public static bool operator ==(UInt16 x, Integer y) { return x == y.value; }
        public static bool operator !=(UInt16 x, Integer y) { return x != y.value; }
        public static bool operator ==(UInt32 x, Integer y) { return x == y.value; }
        public static bool operator !=(UInt32 x, Integer y) { return x != y.value; }
        public static bool operator ==(UInt64 x, Integer y) { return x == y.value; }
        public static bool operator !=(UInt64 x, Integer y) { return x != y.value; }
        public static bool operator >(Int16 x, Integer y) { return x > y.value; }
        public static bool operator <(Int16 x, Integer y) { return x < y.value; }
        public static bool operator >(Int32 x, Integer y) { return x > y.value; }
        public static bool operator <(Int32 x, Integer y) { return x < y.value; }
        public static bool operator >(Int64 x, Integer y) { return x > y.value; }
        public static bool operator <(Int64 x, Integer y) { return x < y.value; }
        public static bool operator >(UInt16 x, Integer y) { return x > y.value; }
        public static bool operator <(UInt16 x, Integer y) { return x < y.value; }
        public static bool operator >(UInt32 x, Integer y) { return x > y.value; }
        public static bool operator <(UInt32 x, Integer y) { return x < y.value; }
        public static bool operator >(UInt64 x, Integer y) { return x > y.value; }
        public static bool operator <(UInt64 x, Integer y) { return x < y.value; }
        public static bool operator >=(Int16 x, Integer y) { return x >= y.value; }
        public static bool operator <=(Int16 x, Integer y) { return x <= y.value; }
        public static bool operator >=(Int32 x, Integer y) { return x >= y.value; }
        public static bool operator <=(Int32 x, Integer y) { return x <= y.value; }
        public static bool operator >=(Int64 x, Integer y) { return x >= y.value; }
        public static bool operator <=(Int64 x, Integer y) { return x <= y.value; }
        public static bool operator >=(UInt16 x, Integer y) { return x >= y.value; }
        public static bool operator <=(UInt16 x, Integer y) { return x <= y.value; }
        public static bool operator >=(UInt32 x, Integer y) { return x >= y.value; }
        public static bool operator <=(UInt32 x, Integer y) { return x <= y.value; }
        public static bool operator >=(UInt64 x, Integer y) { return x >= y.value; }
        public static bool operator <=(UInt64 x, Integer y) { return x <= y.value; }
    }
    public static class Program
    {
        private static bool IntegerFunction(Integer n)
        {
            //code that implements IntegerFunction goes here
            //note that there is NO code that checks the type of n in rum time, because it is NOT needed anymore 
        }
        private static void Main()
        {
            Console.WriteLine("{0}",IntegerFunction(0)); //compile error: there is no overloaded METHOD for objects of type "int" and no implicit conversion from any object, including "int", to "Integer" is known.
            Console.WriteLine("{0}",IntegerFunction(new Integer(0))); //both compiles and no run time error
            Console.WriteLine("{0}",IntegerFunction("string")); //compile error: there is no overloaded METHOD for objects of type "string" and no implicit conversion from any object, including "string", to "Integer" is known.
            Console.WriteLine("{0}",IntegerFunction(new Integer("string"))); //compile error: there is no overloaded CONSTRUCTOR for objects of type "string"
        }
    }
    

    Note that in order to use dynamic in your code you must Add Reference to Microsoft.CSharp

    If the version of the .NET framework is below/under/lesser than 4.0 and dynamic is undefined in that version then you will have to use object instead and do casting to the integer type, which is trouble, so I recommend that you use at least .NET 4.0 or newer if you can so you can use dynamic instead of object.

提交回复
热议问题