gpt4 book ai didi

c# - 如何使动态加载的程序集中的设置提供程序可用于反射?

转载 作者:太空狗 更新时间:2023-10-29 20:39:54 30 4
gpt4 key购买 nike

正在关注 this question ,我在我们正在开发的遗留 C# 应用程序中成功创建了我的自定义设置提供程序。它通过 SettingsProvider 属性引用:

public sealed class MySettings : SettingsProvider
{
...
}

[SettingsProvider(typeof(MySettings))]
internal sealed partial class Settings {}

但是,现在我遇到了另一个问题。

我们的客户端应用程序包含一个自动更新工具,它的实现是为了将大部分客户端(包括上面的类)内置到一个 DLL 中(我们在这里称之为 client.dll),然后由 EXE 使用。 EXE 首先检查更新并在需要时从更新服务器下载最新的更新,用更新的版本(包括 client.dll)替换所有 DLL 等。为了能够在运行时替换 DLL,它不能静态链接到它们。所以在更新之后,它会加载 client.dll 并像这样运行它:

Assembly assy = Assembly.LoadFile(
AppDomain.CurrentDomain.BaseDirectory + "client.dll");
object frm = assy.CreateInstance("Client.Forms.MainForm");
Application.Run((Form)frm);

不幸的是,框架无法在动态加载的程序集中找到我的自定义设置提供程序类。我尝试使用 LoadFrom 而不是上面的 LoadFile 但它没有帮助。到目前为止我找到的唯一可行的解​​决方案是在加载程序 exe 中实现一个与真实设置提供程序同名的代理类,框架可以找到它。然后代理实例化来自客户端程序集的真实设置提供程序并将所有调用委托(delegate)给它。

这似乎可行,但我对此并不满意。有没有办法帮助框架直接在动态加载的程序集中找到我的类?

更新

我得到的错误信息:

System.Configuration.ConfigurationErrorsException: Failed to load provider type: Client.Properties.MySettings, Client, Version=4.0.1341.0, Culture=neutral, PublicKeyToken=null.
at System.Configuration.ApplicationSettingsBase.get_Initializer()
at System.Configuration.ApplicationSettingsBase.CreateSetting(PropertyInfo propInfo)
at System.Configuration.ApplicationSettingsBase.EnsureInitialized()
at System.Configuration.ApplicationSettingsBase.get_Properties()
at System.Configuration.SettingsBase.GetPropertyValueByName(String propertyName)
at System.Configuration.SettingsBase.get_Item(String propertyName)
at System.Configuration.ApplicationSettingsBase.GetPropertyValue(String propertyName)
at System.Configuration.ApplicationSettingsBase.get_Item(String propertyName)
at Client.Properties.Settings.get_SomeConfigSetting()
at ...

通过调试和日志消息,我确定该类的初始化方法从未被调用,因此该类从未被实例化。也就是说,上述异常的背后似乎并没有隐藏的初始化错误。

还有一点可能很重要:客户端目前在 .NET 2.0 上运行,并且在可预见的 future 没有升级的计划。

更新2

我开始按照@jwddixon 的回答建议调查 AppDomain。首先,我想检查 Client.dll 是否真的位于与调用方 EXE 不同的应用程序域中。所以我列出了当前应用程序域中的程序集,并看到 Client 实际上就在那里。但令我惊讶的是,我注意到列表中实际上有两个名为 Client 的程序集 - EXE 和 DLL 具有相同的程序集名称,但版本不同(EXE 为 4.0.0.0,4.0 .1352.0 目前为 DLL)。到目前为止我还没有完全意识到这一点,这可能很重要。接下来我将尝试更改 EXE 程序集名称...

更新3

...这实际上解决了问题! Aaargh...对于我不知道为什么发明了这个扭曲方案的无名前辈,此刻我有非常不愉快的想法...但是感谢你们所有人提出的问题和想法,最终导致了解决方案!

最佳答案

我还没有尝试过这个,但是像这样的东西怎么样:

AppDomainSetup setup = AppDomain.CurrentDomain.SetupInformation;
AppDomain domain = AppDomain.CreateDomain("myAppDomain", null, setup)
setup.ApplicationBase = file;

Assembly assy = domain.Load(AssemblyName.GetAssemblyName(
AppDomain.CurrentDomain.BaseDirectory +
"client.dll"));

object frm = assy.CreateInstance("Client.Forms.MainForm");
Application.Run((Form)frm);

关于c# - 如何使动态加载的程序集中的设置提供程序可用于反射?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15153420/

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