从子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)中的原因.
解决方法:
首先,您应该使用共享代码创建一个新程序集.特别是,您应该为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.
标签:appdomain,c,net 来源: https://codeday.me/bug/20191031/1976084.html