Подтвердить что ты не робот

С# - Ссылка на тип в динамически сгенерированной сборке

Я пытаюсь выяснить, возможно ли это, когда вы динамически генерируете сборки, ссылаться на тип в ранее динамически сгенерированной сборке.

Например:

using System;
using System.CodeDom.Compiler;
using System.Reflection;
using Microsoft.CSharp;

CodeDomProvider provider = new CSharpCodeProvider();
CompilerParameters parameters = new CompilerParameters();

parameters.GenerateInMemory = true;

CompilerResults results = provider.CompileAssemblyFromSource(parameters, @"
namespace Dynamic
{
    public class A
    {
    }
}
");

Assembly assem = results.CompiledAssembly;

CodeDomProvider provider2 = new CSharpCodeProvider();
CompilerParameters parameters2 = new CompilerParameters();

parameters2.ReferencedAssemblies.Add(assem.FullName);
parameters2.GenerateInMemory = true;

CompilerResults results2 = provider2.CompileAssemblyFromSource(parameters2, @"
namespace Dynamic
{
    public class B : A
    {
    }
}
");

if (results2.Errors.HasErrors)
{
    foreach (CompilerError error in results2.Errors)
    {
        Console.WriteLine(error.ErrorText);
    }
}
else
{
    Assembly assem2 = results2.CompiledAssembly;
}

Этот код выводит на консоль следующие команды: The type or namespace name 'A' could not be found (are you missing a using directive or an assembly reference?)

Я пробовал много разных способов, но ничего не работает. Я что-то упускаю? Возможно ли это?

EDIT: исправление ошибки в коде приводит к этой ошибке: Metadata file 'l0livsmn, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null' could not be found

EDIT2: бит побочной заметки, но изменение GenerateInMemory на false, и выполнение parameters2.ReferencedAssemblies.Add(assem.Location); приведет к его правильной компиляции, но я бы предпочел ссылаться на сборку, которая находится непосредственно в памяти, а не выводить временные файлы.

4b9b3361

Ответ 1

Я думаю, что в

CompilerResults results2 = provider2.CompileAssemblyFromSource(parameters, @"
namespace Dynamic
{
    public class B : A
    {
    }
}
");

Вы хотите передать parameters2, а не parameters.

Я нашел способ сделать это, вам не нужно компилировать первый в памяти, если вы этого не сделаете, он создаст dll для этой сборки в вашем временном каталоге, а также при вызове

ReferencedAssemblies.Add() 

вы не передадите имя сборки, вы пройдете путь сборки, посмотрите на этот код, он должен работать безупречно:

        CodeDomProvider provider = new CSharpCodeProvider();
        CompilerParameters parameters = new CompilerParameters();            

        CompilerResults results = provider.CompileAssemblyFromSource(parameters, @"
            namespace Dynamic
            {
                public class A
                {
                }
            }
            ");

        Assembly assem = results.CompiledAssembly;

        CodeDomProvider provider2 = new CSharpCodeProvider();
        CompilerParameters parameters2 = new CompilerParameters();

        parameters2.ReferencedAssemblies.Add(assem.Location);
        parameters2.GenerateInMemory = true;

        CompilerResults results2 = provider2.CompileAssemblyFromSource(parameters2, @"
            namespace Dynamic
            {
                public class B : A
                {
                }
            }
            ");

        if (results2.Errors.HasErrors)
        {
            foreach (CompilerError error in results2.Errors)
            {
                Console.WriteLine(error.ErrorText);
            }
        }
        else
        {
            Assembly assem2 = results2.CompiledAssembly;
        }

Ответ 2

MSDN говорит, что вы можете:

Ограничения по типам ссылок

Ассембли могут ссылаться на определенные типы в другой сборке. Переходный динамическая сборка может безопасно ссылаться типы, определенные в другом переходном динамическая сборка, устойчивая динамической сборки или статической сборка. Однако общие язык исполнения не позволяет устойчивый динамический модуль для укажите тип, определенный в переходный динамический модуль. Это потому что когда сохраняется динамическая модуль загружается после сохранения в диска, среда выполнения не может ссылки на типы, определенные в переходный динамический модуль.