我有一个带有自定义类型转换器的简单类型,它在运行时编译和加载。但是,TypeDescriptor.GetConverter()并没有找到合适的转换器。这是一个独立的例子
using System;
using System.ComponentModel;
using System.Collections.Generic;
using System.CodeDom.Compiler;
using Microsoft.CSharp;
public class Program
{
private static string src =
@"
using System;
using System.ComponentModel;
namespace LoadMe
{
[TypeConverter(typeof(FooConverter))]
public class Foo
{
}
public class FooConverter : TypeConverter
{
// stuff
}
}
";
public static void Main()
{
var codeProvider = new CSharpCodeProvider(new Dictionary<string, string>{{ "CompilerVersion", "v4.0" }});
var compileParameters = new CompilerParameters(new[] { "System.dll" }) { GenerateInMemory = true };
var compilerResults = codeProvider.CompileAssemblyFromSource(compileParameters, src);
if (compilerResults.Errors.Count == 0)
{
var fooType = compilerResults.CompiledAssembly.GetType("LoadMe.Foo");
Console.WriteLine(fooType.FullName + "::" + fooType.Assembly.FullName);
Console.WriteLine("Type converter type = '" + TypeDescriptor.GetConverter(fooType).GetType().FullName + "'");
}
else
{
foreach (var err in compilerResults.Errors)
Console.WriteLine(err);
}
}
}
以下是示例的输出(在vs中编译)
LoadMe.Foo::q5sszdls, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null
Type converter type = 'System.ComponentModel.TypeConverter'
我应该自己挖掘属性并自己创建实例,还是我在这里遗漏了其他东西?
也很奇怪!当在LINQPad中作为C#程序运行时,它确实有效!这是输出。 LINQPad ready source
LoadMe.Foo::oqmid5in, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null
Type converter type = 'LoadMe.FooConverter'
知道LINQPad实际上有什么不同才能让它找到合适的转换器吗?
答案 0 :(得分:10)
.NET使用反射来加载指定的类型转换器。当您将Type
实例传递给TypeConverterAttribute
时,您的TypeConverterAttribute
将包含一个程序集限定的类型名称。加载该程序集限定的类型名称不起作用,因为无法找到程序集。 (是的,即使已经加载了该程序集。)
您可以使用构造函数重载,它接受一个字符串来指定类型名称,而不引用定义程序集,在这种情况下,它将在与您要求的转换器类型相同的程序集中查找:
[TypeConverter("LoadMe.FooConverter")]
public class Foo
{
}