Given a constrained generic method can I call a non-generic method passing the actual type of the generic parameter

后端 未结 2 706
借酒劲吻你
借酒劲吻你 2021-01-21 21:25

Awkward title I know, this is best explained in code. Given a set of classes:

public abstract class MyBaseType
{
    public string Message { get; set; }
}

publi         


        
相关标签:
2条回答
  • 2021-01-21 22:20

    As Aliostad said, what your trying to do is no longer generic and it would be better to just use overloads. It looks like you are trying to do something similar to template specialization in C++, where depending on the generic type it calls different methods.

    Here is an example where I implemented a sort of generic specialization using reflection, you might be able to apply a similar pattern if overloading method wont work for you. If you can cache the results of reflection and only call GetMethod once then it turns out to be not too slow. Inside a class generic by T there is a method that calls:

    if (_serializeDataToStream == null)
        _serializeDataToStream = (Action<BinaryWriter, int, T[]>)GetTypeSpecificSerializationMethod();
    
    _serializeDataToStream(writer, _size, _data);
    

    Where the GetTypeSpecific method uses reflection to create a delegate

    /// <summary>
    /// Returns a delegate that points at the static type specific serialization method
    /// </summary>
    /// <returns></returns>
    private Delegate GetTypeSpecificDeserializationMethod()
    {
        if (typeof(T) == typeof(double))
        {
            MethodInfo method = this.GetType().GetMethod("DeserializeDouble", BindingFlags.Static | BindingFlags.NonPublic);
            return Delegate.CreateDelegate(typeof(Action<BinaryReader, int, T[]>), method);
        }
        else if (typeof(T) == typeof(ushort))
        {
            MethodInfo method = this.GetType().GetMethod("DeserializeUshort", BindingFlags.Static | BindingFlags.NonPublic);
            return Delegate.CreateDelegate(typeof(Action<BinaryReader, int, T[]>), method);
        }
        else if (typeof(T) == typeof(DateTime))
        {
            MethodInfo method = this.GetType().GetMethod("DeserializeDateTime", BindingFlags.Static | BindingFlags.NonPublic);
            return Delegate.CreateDelegate(typeof(Action<BinaryReader, int, T[]>), method);
        }
        else if (typeof(T) == typeof(bool))
        {
            MethodInfo method = this.GetType().GetMethod("DeserializeBool", BindingFlags.Static | BindingFlags.NonPublic);
            return Delegate.CreateDelegate(typeof(Action<BinaryReader, int, T[]>), method);
        }
    
        throw new NotImplementedException("No deserialization method has been setup for type " + typeof(T).FullName);
    }
    
    /// <summary>
    /// Serialize double[] to BinaryWriter
    /// </summary>
    /// <param name="writer"></param>
    /// <param name="size"></param>
    /// <param name="data"></param>
    private static void SerializeDouble(BinaryWriter writer, int size, double[] data)
    {
        for (int i = 0; i < size; i++)
        {
            writer.Write(data[i]);
        }
    }
    
    0 讨论(0)
  • 2021-01-21 22:20

    The problem here is you are not really expressing a generality. You have to implement OutputFormattedTypeInfo for each and every base type, so you might as well forget about the generic method and simply use the overloads (you do not need generics here):

    public void OutputFormattedTypeInfo(BaseType theType)
    {
        // ...
    }
    
    public void OutputFormattedTypeInfo(MySuperType theType)
    {
        // ...
    }
    
    public void OutputFormattedTypeInfo(MyOtherSuperType theType)
    {
        // ...
    }
    
    0 讨论(0)
提交回复
热议问题