3

アプリケーションを管理するためのツールを開発しようとしています。ツールは、具体的なアプリから特定の情報を知る必要があります。これらのアプリケーションによって参照されるすべてのアセンブリを参照せずに、そのデータを取得したいと考えています。
これは私が持っているコードです:

MetadataLoader.dll:

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

    public Metadata Metadata { get; set; }
}

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

メタデータ.dll:

public class Metadata 
{
    ...
}

ツール.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) にロードしたいと考えています。

4

1 に答える 1

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異なる AppDomains からこのクラスのインスタンスにアクセスできるようにするには、から継承する必要があることに注意してください。

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 は、.dll を含むサードパーティの DLL のみを参照しますMetadataSource
  • MetadataLoader.dll と Tool.dll は、サードパーティの DLL と Shared.dll を参照します。
于 2012-12-21T20:33:26.100 に答える