从子AppDomain检索数据而不从父域引用程序集

时间:2012-12-21 16:00:27

标签: c# .net appdomain

我正在尝试开发用于管理我的应用程序的工具。工具需要知道具体应用程序的某些信息。我希望在不引用这些应用程序引用的所有程序集的情况下检索该数据 这是我的代码:

MetadataLoader.dll:

public class MetadataLoader 
{
    public MetadataLoader(MetadataSource source) 
    {
        ...
    }

    public Metadata Metadata { get; set; }
}

public class MetadataSource 
{
    public void Load(string path) 
    {
        ...
    }
}

Metadata.dll:

public class Metadata 
{
    ...
}

Tool.dll:

public class Worker 
{
    public void Execute() 
    {
        // create new domain
        // create metadataloader instance
        //  - create metadatasource instance and call Load. Path should be passed from this method. 
        //  - create MetadataLoader instance 
        //  - fetch MetadataLoader.Metadata
        // >>> Tool.dll should reference only Metadata.dll, not MetadataLoader.dll
    }
}

此外,我无法更改MetadataSource类,因为它位于第三方库中(这是简化示例)。 我该如何实现Execute方法?

更新 MetadataLoader.dll和Tool.dll引用相同程序集的不同版本(初始问题中提到的第三方库),因此Tool.dll不得引用MetadataLoader.dll使用的版本。

Tool.dll引用: - ThirdParty.v2.dll - Metadata.dll

MetadataLoader.dll引用: - ThirdParty.v1.dll - Metadata.dll

为此问题提供更多背景信息。 Tool.dll和MetadataLoader.dll都使用相同的ORM来访问数据库。 ORM还提供对象模型元数据。 Tool.dll使用ORM访问自己的数据库,但也需要访问受管应用程序的对象模型。 Tool.dll不得依赖于管理的应用程序使用的ORM版本。这就是为什么我想将管理的应用程序加载到自己的“容器”(AppDomain)中。

1 个答案:

答案 0 :(得分:4)

首先,您应该使用共享代码创建一个新程序集。特别是,您应该为MetadataSource创建一个包装器。这是必要的,因为MetadataSource类是在第三方库中定义的,您无法直接将其更改为从MarshalByRefObject继承。

public class MetadataSourceWrapper : MarshalByRefObject
{
    public MetadataSource MetadataSource { get; private set; }

    public MetadataSourceWrapper()
    {
        MetadataSource = new MetadataSource();
    }

    public void Load(string path)
    {
        MetadataSource.Load(path);
    }
}

请注意,它应该从MarshalByRefObject继承,以允许您从不同的AppDomain访问此类的实例。

您还应该按如下方式定义IMetadataLoader接口:

public interface IMetadataLoader
{
    void Init(MetadataSourceWrapper source);
    Metadata Metadata { get; set; }
}

之后,将MetadataLoader类更改为继承MarshalByRefObject并实施IMetadataLoader,如下所示:

public class MetadataLoader : MarshalByRefObject, IMetadataLoader
{
    public MetadataLoader() { }

    public void Init(MetadataSourceWrapper source)
    {
        ...
    }

    public Metadata Metadata { get; set; }
}

最后实现GetMetadata方法:

public static Metadata GetMetadata(
    string metadataLoaderApplicationBase, /*e.g. `C:\MyCompany\MyApp*`*/
    string metadataLoaderAssemblyName, /*e.g. `MetadataLoader, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null` or `MetadataLoader`*/
    string sharedDllAssemblyName, /*e.g. `Shared, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null` or `Shared`*/
    string metadataSourcePath
)
{
    AppDomainSetup domainSetup = new AppDomainSetup();
    domainSetup.ApplicationBase = Path.GetDirectoryName(assemblyName);

    Evidence evidence = AppDomain.CurrentDomain.Evidence;

    AppDomain newDomain = AppDomain.CreateDomain("AppDomain Friendly Name",
        evidence, domainSetup);

    MetadataSourceWrapper msw =
        (MetadataSourceWrapper)newDomain.CreateInstanceAndUnwrap(sharedDllAssemblyName,
            "YourNamespace.MetadataSourceWrapper" /*full type name including the namespace*/);
    msw.Load(metadataSourcePath);

    IMetadataLoader metadataLoader =
        (IMetadataLoader)newDomain.CreateInstanceAndUnwrap(metadataLoaderAssemblyName,
            "YourNamespace.MetadataLoader" /*full type name including the namespace*/);
    metadataLoader.Init(msw);

    Metadata metadata = metadataLoader.Metadata;
    AppDomain.Unload(newDomain);
    return metadata;
}

关于参考文献:

  • Shared.dll将仅引用包含MetadataSource
  • 的第三方DLL
  • MetadataLoader.dll和Tool.dll将引用第三方DLL和Shared.dll。