我正在尝试动态更改程序集绑定(从一个版本到另一个版本)。
我已经尝试过这段代码,但它不起作用:
Configuration config = ConfigurationManager.OpenExeConfiguration(ConfigurationUserLevel.None);
ConfigurationSection assemblyBindingSection = config.Sections["assemblyBinding"];
assemblyBindingSection.SectionInformation.ConfigSource = "bindingConf.xml";
config.Save(ConfigurationSaveMode.Modified);
ConfigurationManager.RefreshSection("assemblyBinding");
bindingConf.xml
包含assemblyBinding部分配置。
那么可以在运行时更改此部分吗?怎么做?我有哪些替代方案?
答案 0 :(得分:19)
我发现动态绑定到程序集的不同版本的最好方法是挂钩AppDomain.AssemblyResolve
事件。只要运行时无法找到应用程序链接的确切程序集,就会触发此事件,并且它允许您在其位置提供另一个自己加载的程序集(只要它是兼容的)。
例如,您可以在应用程序的主类上放置一个静态构造函数来挂钩事件,如下所示:
using System.Reflection;
static Program()
{
AppDomain.CurrentDomain.AssemblyResolve += delegate(object sender, ResolveEventArgs e)
{
AssemblyName requestedName = new AssemblyName(e.Name);
if (requestedName.Name == "AssemblyNameToRedirect")
{
// Put code here to load whatever version of the assembly you actually have
return Assembly.LoadFrom("RedirectedAssembly.DLL");
}
else
{
return null;
}
};
}
此方法避免了处理配置文件中的程序集绑定的需要,并且就您可以使用它做什么而言更灵活。
答案 1 :(得分:3)
使用以下代码在运行时更新配置文件的RuntimeSection:
private void ModifyRuntimeAppConfig()
{
XmlDocument modifiedRuntimeSection = GetResource("Framework35Rebinding");
if(modifiedRuntimeSection != null)
{
Configuration config = ConfigurationManager.OpenExeConfiguration(ConfigurationUserLevel.None);
ConfigurationSection assemblyBindingSection = config.Sections["runtime"];
assemblyBindingSection.SectionInformation.SetRawXml(modifiedRuntimeSection.InnerXml);
config.Save(ConfigurationSaveMode.Modified);
ConfigurationManager.RefreshSection("runtime");
}
}
包含Framework35Rebinding:
<runtime>
<assemblyBinding xmlns="urn:schemas-microsoft-com:asm.v1">
<dependentAssembly>
<assemblyIdentity name="Microsoft.Build.Framework" publicKeyToken="b03f5f7f11d50a3a" culture="neutral"/>
<bindingRedirect oldVersion="0.0.0.0-99.9.9.9" newVersion="3.5.0.0"/>
</dependentAssembly>
<dependentAssembly>
<assemblyIdentity name="Microsoft.CompactFramework.Build.Tasks" publicKeyToken="b03f5f7f11d50a3a" culture="neutral"/>
<bindingRedirect oldVersion="0.0.0.0-99.9.9.9" newVersion="9.0.0.0"/>
</dependentAssembly>
</assemblyBinding>
</runtime>
和包含(在执行程序之前)的app.config:
<?xml version="1.0"?>
<configuration>
<startup>
<supportedRuntime version="v2.0.50727"/>
</startup>
<runtime>
</runtime>
</configuration>
尽管如此,它不能用于我想做的事情,因为仅在应用程序启动时读取assemblyBinding,而RefreshSection("runtime")
答案 2 :(得分:1)
我喜欢埃里克的回答。尝试在Web应用程序中使用新的越野车NuGet PackageReference模型时,这是一种救命稻草。问题是您可以让msbuild自动生成绑定,但是,它们生成到Assembly.dll.config而不是web.config的绑定。因此,这种解决方法很棒。
我对Eric的代码进行了一些修改,使其更加通用并可以与ASP.Net Core应用程序一起使用:
AppDomain.CurrentDomain.AssemblyResolve += delegate (object sender2, ResolveEventArgs e2)
{
var requestedNameAssembly = new AssemblyName(e2.Name);
var requestedName = requestedNameAssembly.Name;
if (requestedName.EndsWith(".resources")) return null;
var binFolder = System.Web.Hosting.HostingEnvironment.MapPath("~/bin");
var fullPath = Path.Combine(binFolder, requestedName) + ".dll";
if (File.Exists(fullPath))
{
return Assembly.LoadFrom(fullPath);
}
return null;
};