Ho un pratico metodo di utilità che prende il codice e sputa un assembly in memoria. (Usa CSharpCodeProvider
, anche se io non credo che dovrebbe importa.) Questo gruppo funziona come qualsiasi altro con la riflessione, ma quando viene utilizzato con la parola chiave dynamic
, sembra non riuscire con un RuntimeBinderException
:Il tentativo di associare un metodo dinamico a un assieme creato in modo dinamico causa una RuntimeBinderException
'oggetto' non contiene una definizione per 'Sound'
Esempio:
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
qualcuno sa il motivo per cui la DLR, non è in grado di gestire questa situazione? C'è qualcosa che potrebbe essere fatto per risolvere questo scenario?
EDIT:
metodo CreateAssembly:
responsabilità: alcune di queste cose contiene i metodi di estensione, i tipi personalizzati, ecc dovrebbe essere auto-esplicativo però.
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;
}
Errore stupido da parte mia. 'dynamic' rispetta l'accessibilità, quindi non può eseguire codice non pubblico da un altro assembly, mentre la riflessione non interessa un po 'dell'accessibilità. Grazie. – MgSam