无法在运行时强制转换对象

时间:2015-05-21 13:11:52

标签: c# asp.net-mvc

我正在学习ASP.NET MVC 5(vNext)。为了做到这一点,我正在迁移现有的应用程序。在该应用程序中,我获得了一个实现特定接口的类列表。为了做到这一点,我使用以下代码:

// Find all classes that implement IMyInterface
var type = typeof(IMyInterface);
var classes = AppDomain.CurrentDomain.GetAssemblies()
                .SelectMany(x => x.GetTypes())
                .Where(y => type.IsAssignableFrom(y) && y.GetInterfaces().Contains(type))
                .ToList();

if (classes == null)
  Console.WriteLine("None found");
else            
  Console.WriteLine(classes.Count + " found."); 

try {
  foreach (var c in classes)
  {
    Console.WriteLine(c.GetType().FullName);
    var converted = (IMyInterface)(c);
    // Never gets here. Exception gets thrown.  
  }
}
catch (Exception ex)
{
  Console.WriteLine(ex.Message);
  // Prints: Cannot cast from source type to destination type.
}       

不幸的是,抛出的异常是:“无法从源类型转换为目标类型。”当我打印出该类型的全名时,它是System.MonoType。我做错了什么?

1 个答案:

答案 0 :(得分:7)

在您的代码中,classesList<Type>,也就是说这些不是您的类的实例,它们是类Type的实例,它描述了实现您的接口的类

因此这一行

var converted = (IMyInterface)(c);

总是会抛出异常,因为Type没有实现IMyInterface。我怀疑你真正想要做的是使用Type 实例化你的类的一个实例,这可以通过Activator上的静态方法实现,例如

var converted = (IMyInterface)Activator.CreateInstance(c);

扩展我对您提出的问题的评论

if (classes == null)

以上行永远不会评估为真,您可能想检查列表是否为空

if (classes.Count == 0)

或者,实际上您根本不需要列表,请删除.ToList()并考虑使用

// classes now an IEnumerable<Type> - no need for a list here.
var classes = AppDomain.CurrentDomain.GetAssemblies()
            .SelectMany(x => x.GetTypes())
            .Where(y => type.IsAssignableFrom(y) && y.GetInterfaces().Contains(type));
if (!classes.Any())