コードを取り込んでメモリ内アセンブリを吐き出す便利なユーティリティメソッドがあります。このアセンブリは、反射して、他のように動作します(私はそれは問題べきではないと思うが、それは、CSharpCodeProvider
を使用しています。)が、dynamic
キーワードで使用した場合、RuntimeBinderException
で失敗しているようだ:動的に作成されたアセンブリで動的メソッドをバインドしようとすると、RuntimeBinderExceptionが発生します。
「オブジェクト」 「サウンド」の定義が含まれていません
例:
var assembly = createAssembly("class Dog { public string Sound() { return \"woof\"; } }");
var type = assembly.GetType("Dog");
Object dog = Activator.CreateInstance(type);
var method = type.GetMethod("Sound");
var test1Result = method.Invoke(dog, null); //This returns "woof", as you'd expect
dynamic dog2 = dog;
String test2Result = dog2.Sound(); //This throws a RuntimeBinderException
誰もがDLRがこれを処理することができない理由を知っていますか?このシナリオを修正するためにできることはありますか?
EDIT:
createAssembly方法:
免責事項:このようなもののいくつかはそれはしかし、自己説明的であるべき拡張メソッド、カスタムタイプなどが含まれています。
private Assembly createAssembly(String source, IEnumerable<String> assembliesToReference = null)
{
//Create compiler
var codeProvider = new CSharpCodeProvider();
//Set compiler parameters
var compilerParameters = new CompilerParameters
{
GenerateInMemory = true,
GenerateExecutable = false,
CompilerOptions = "/optimize",
};
//Get the name of the current assembly and everything it references
if (assembliesToReference == null)
{
var executingAssembly = Assembly.GetExecutingAssembly();
assembliesToReference = executingAssembly
.AsEnumerable()
.Concat(
executingAssembly
.GetReferencedAssemblies()
.Select(a => Assembly.Load(a))
)
.Select(a => a.Location);
}//End if
compilerParameters.ReferencedAssemblies.AddRange(assembliesToReference.ToArray());
//Compile code
var compilerResults = codeProvider.CompileAssemblyFromSource(compilerParameters, source);
//Throw errors
if (compilerResults.Errors.Count != 0)
{
throw new CompilationException(compilerResults.Errors);
}
return compilerResults.CompiledAssembly;
}
私の部分では愚かな間違いです。 'dynamic'はアクセシビリティを尊重しているので、別のアセンブリからの非公開コードを実行することはできません。一方、リフレクションはアクセシビリティについては気にしません。ありがとう。 – MgSam