如何在运行时更新配置文件中的assemblyBinding部分?

时间:2008-11-28 13:35:39

标签: c# .net configuration

我正在尝试动态更改程序集绑定(从一个版本到另一个版本)。

我已经尝试过这段代码,但它不起作用:

      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部分配置。

那么可以在运行时更改此部分吗?怎么做?我有哪些替代方案?

3 个答案:

答案 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;
            };