Я работаю над проектом, который будет использовать CodeDOM для создания класса, который оценивает пользовательское выражение, создает сборку для класса и загружает сборку.Поскольку может быть довольно много пользовательских выражений, я хотел бы сначала создать домен приложения, выполнить создание / загрузку и выполнение CodeDOM для сборки в этом домене приложения, а затем выгрузить домен приложения.
IЯ немного искал и нашел много примеров того, как загрузить существующую сборку в домен приложений, но я не могу найти тот, который показывает мне, как создать сборку из внутри домена приложений.
В этом примере ( DynamicCode ) создается сборка с использованием CodeDOM, а затем загружается ее в домен приложений, однако автор генерирует сборку на диск.Я бы предпочел сгенерировать сборку в памяти, чтобы мне не приходилось управлять очисткой сгенерированных сборок.(хотя при этом создается папка .dll во временной папке).
Может кто-нибудь указать мне пример того, как это сделать?
Любая помощь будет принята с благодарностью.
Я включил некоторые выдержки из своего кода, чтобы вы все могли почувствовать то, что у меня есть:
private string CreateSource()
{
CodeCompileUnit codeUnit = new CodeCompileUnit();
CodeNamespace codeNamespace = new CodeNamespace(Namespace);
CodeTypeDeclaration codeClass = new CodeTypeDeclaration
{
Name = "ExpressionEvaluator",
IsClass = true,
TypeAttributes = TypeAttributes.Public | TypeAttributes.Sealed
};
codeNamespace.Types.Add(codeClass);
codeUnit.Namespaces.Add(codeNamespace);
AddMethods(codeClass);
string result = GenerateSourceCode(codeUnit);
return result.ToString();
}
private CompilerResults CompileSource(string source)
{
using (CodeDomProvider provider = new CSharpCodeProvider())
{
CompilerParameters parameters = CreateCompilerParameters();
CompilerResults result = CompileCode(provider, parameters, source);
return result;
}
}
private static CompilerParameters CreateCompilerParameters()
{
CompilerParameters result = new CompilerParameters
{
CompilerOptions = "/target:library",
GenerateExecutable = false,
GenerateInMemory = true
};
result.ReferencedAssemblies.Add("System.dll");
return result;
}
private object RunEvaluator(CompilerResults compilerResults)
{
object result = null;
Assembly assembly = compilerResults.CompiledAssembly;
if (assembly != null)
{
string className = "ExpressionEvaluator";
object instance = assembly.CreateInstance("Lab.ExpressionEvaluator");
Module[] modules = assembly.GetModules(false);
Type type = (from t in modules[0].GetTypes()
where t.Name == className
select t).FirstOrDefault();
MethodInfo method = (from m in type.GetMethods()
where m.Name == "Evaluate"
select m).FirstOrDefault();
result = method.Invoke(instance, null);
}
else
{
throw new Exception("Unable to load Evaluator assembly");
}
return result;
}
Я считаю, что эти фрагменты кода показывают основные функциональные возможности моего проекта.Теперь все, что мне нужно сделать, это обернуть его в свой собственный домен приложений.