gpt4 book ai didi

c# - 在不从父域引用程序集的情况下从子 AppDomain 检索数据

转载 作者:行者123 更新时间:2023-11-30 15:36:09 24 4
gpt4 key购买 nike

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

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- 元数据.dll

MetadataLoader.dll 引用:- ThirdParty.v1.dll- 元数据.dll

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

最佳答案

首先,您应该使用共享代码创建一个新程序集。特别是,您应该为 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 接口(interface):

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。

关于c# - 在不从父域引用程序集的情况下从子 AppDomain 检索数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13993291/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com