3

我正在尝试使用 Activator.CreateInstance 创建一个类并将其转换为它实现的通用接口。

为什么我在运行时收到“InvalidCastException”(.Net 4.0)?

这是代码:

namespace CastingGenerics
{
    class Program
    {
        static void Main(string[] args)
        {
            var typeFromSomewhere = typeof(ActionSomethingValidator);

            // This line throw 'System.InvalidCastException'
            // "Unable to cast object of type 'CastingGenerics.ActionSomethingValidator' to type 'CastingGenerics.IActionValidator`1[CastingGenerics.IAction]'."
            var castingError = (IActionValidator<IAction>)Activator.CreateInstance(typeFromSomewhere);
        }
    }

    internal interface IAction
    {
    }

    internal interface IActionValidator<in T>
        where T : IAction
    {
        bool Validate(T action, out string failure);
    }

    internal abstract class ActionValidator<T> : IActionValidator<T>
        where T : IAction
    {
        public abstract bool Validate(T action, out string failure);
    }

    internal class ActionSomethingValidator : ActionValidator<IActionSomething>
    {
        public override bool Validate(IActionSomething action, out string failure)
        {
            failure = string.Empty;
            return true;
        }
    }

    internal interface IActionSomething : IAction
    {
    }
}
4

2 回答 2

1

你得到的是一个IActionValidator<IActionSomething>. IActionValidator<IActionSomething>不能分配给IActionValidator<IAction>,因为T被标记为in类型参数。

想象一下,对于一个IActionValidator<IAction>名为 的变量v,有以下Validate方法:

Validate(IAction action, out string failure);

现在,如果v指向 的实例IActionValidator<IActionSomething>,您可以通过未实现的IAction实例将实例传递给第一个参数。但是,对于. 因此,编译器不会让你这样做。vIActionSomethingIActionValidator<IActionSomething>

于 2013-03-17T15:39:50.733 回答
0

一种方法是使用动态.

dynamic smth = Activator.CreateInstance(typeFromSomewhere);

由于编辑器无法在设计时确定类型,您将失去自动完成功能。但是您仍然可以调用仅手动编码的方法。 smth.Validate(() => true, "");

于 2020-04-13T21:18:09.497 回答