在WCF程序中动态修改app.config配置文件

今天在个WCF程序中加入了修改配置文件的功能。我是直接通过IO操作修改的app.config文件内容,修改后发现发现其并不生效,用Google搜了一下,在园子里的文章动态修改App.Config 和web.Config中找到了解决方案。

原来,.net framework中对于配置文件不是实时读取的,而是有缓存的。对于那些已经更新了的内容,需要调用ConfigurationManager.RefreshSection(需要添加System.Configuration.dll的引用)函数刷新相应节点。

比较蛋疼的是,这个函数并不支持刷新Group。也就是说,我们不能通过ConfigurationManager.RefreshSection("system.serviceModel")一句话实现对WCF的配置刷新,需要调用如下四句话才行。

ConfigurationManager.RefreshSection("system.serviceModel/behaviors");
ConfigurationManager.RefreshSection("system.serviceModel/bindings");
ConfigurationManager.RefreshSection("system.serviceModel/client");
ConfigurationManager.RefreshSection("system.serviceModel/services");

另外,值得一提的是:如果用IO操作修改修改app.config配置,直接使用相对路径"myapp.exe.config"来修改不可靠的,很容易出现找不到配置文件的异常(原因有很多种),需要使用AppDomain.CurrentDomain.SetupInformation.ConfigurationFile属性来获取配置文件的完整路径。

假设XXX.exe.config内容如下

<?xml version="1.0" encoding="utf-8" ?>
<configuration>
<system.serviceModel>
<bindings>
<basicHttpBinding>
<binding name="ProcessServiceSoap" />
</basicHttpBinding>
</bindings> <client>
<endpoint address="http://192.168.0.123/Services/ProcessService.asmx"
binding="basicHttpBinding" bindingConfiguration="ProcessServiceSoap"
contract="S_ProcessService.ProcessServiceSoap" name="ProcessServiceSoap" />
</client>
</system.serviceModel>
</configuration>

以下方法,修改 endpoint 下指定根据 bindingConfiguration 的值修改 address 。

public void ChangeEndpointAddress(string endpointBindingConfiguration, string address)
{
var config = AppDomain.CurrentDomain.SetupInformation.ConfigurationFile;
XElement root = XElement.Load(config); var quary = from ele in root.Element("system.serviceModel").Element("client").Elements("endpoint") where ele.Attribute("bindingConfiguration").Value == endpointBindingConfiguration select ele;
quary.ElementAt().SetAttributeValue("address", address); //保存上面的修改
root.Save(config);
ConfigurationManager.RefreshSection("system.serviceModel/client");
}

如 ChangeEndpointAddress("ProcessServiceSoap", "http://192.168.111.222/Services/ProcessService.asmx")

将原来的address由192.168.0.123修改为192.168.111.222

首先假设你的应用程序配置文件如下:

<?xml version="1.0" encoding="utf-8" ?>
<configuration>
<appSettings>
<add key="name" value="old"/>
</appSettings>
</configuration>

Ok,那么如何在运行时去修改name的值呢??

有很多童鞋会说可以使用Xml读取配置文件,然后xxx。。。。

当然这种方法肯定可以解决问题,有没有其他方法呢??

在这里我要介绍一种比较简单的方法,可能已经有人知道了,那就是使用ConfigurationManager类

ConfigurationManager 存在System.Configuration.dll 中。

代码如下:

public static void Main()
{
Console.WriteLine(ConfigurationManager.AppSettings["name"]);
ChangeConfiguration();
Console.WriteLine(ConfigurationManager.AppSettings["name"]);
Console.ReadLine();
} private static void ChangeConfiguration()
{
//读取程序集的配置文件
string assemblyConfigFile = Assembly.GetEntryAssembly().Location; Configuration config = ConfigurationManager.OpenExeConfiguration(assemblyConfigFile);
//获取appSettings节点
AppSettingsSection appSettings = (AppSettingsSection)config.GetSection("appSettings"); //删除name,然后添加新值
appSettings.Settings.Remove("name");
appSettings.Settings.Add("name", "new"); //保存配置文件
config.Save();
}

代码很简单:首先读取配置文件,接着获取appSettings节点,然后修改,接着保存。 运行:结果如下:

在WCF程序中动态修改app.config配置文件

可以看到输出的值是两个old.

为什么??

查找msdn文档可以发现微软出于性能考虑,对ConfigurationManager采用了缓存策略,所以如果要读取新的值,应该使用ConfigurationManager的RefreshSection来进行刷新,

ConfigurationManager . RefreshSection:

刷新命名节,这样在下次检索它时将从磁盘重新读取它。

于是将Main方法修改为:

Console.WriteLine(ConfigurationManager.AppSettings["name"]);

ChangeConfiguration();

ConfigurationManager.RefreshSection("appSettings");

Console.WriteLine(ConfigurationManager.AppSettings["name"]);

重新清理解决方案,重新运行:

在WCF程序中动态修改app.config配置文件

可以看到,仍然是两个old。。。

为什么?? 

难道值没有修改??,我们打开应用程序的配置文件,可以通过监视assemblyConfigFile获得路径

上面是xxx\bin\Debug\CAStudy.exe.,对应的配置文件就是CAStudy.exe.config

在WCF程序中动态修改app.config配置文件

文件的内容如下:

在WCF程序中动态修改app.config配置文件

可以发现value 值已经更改,那么为什么输出还是old,old 呢??

为了验证不是VS2010的问题。

首先手动将CAStudy.exe.config 文件中的value改为”old”,接着再次运行CAStudy.exe 结果如下:

在WCF程序中动态修改app.config配置文件

可以看到输出时old,和new。为什么会这样???

难道调试时读取的不是修改的配置文件,或者修改的配置文件并不是调试的应用程序读取的文件??

在assemblyConfigFile 中设置断点,可以发现assemblyConfigFile 读取的是CAStudy.exe.Config。但是vs调试的时候运行的是CAStudy.vshost.exe。也就是说我们使用ConfigurationManager.OpenExeConfiguration 打开的是CAStudy.exe.config文件,但是我们调试的应用程序CAStudy.vshost.exe使用的是CAStudy.vshost.exe.config文件。

那么还有其他的方式可以准确的获取应用程序配置文件吗??

有的,使用AppDomain.CurrentDomain.SetupInformation.ConfigurationFile;

将ChangeConfiguration()方法修改如下:

public static void Main()
{
Console.WriteLine(ConfigurationManager.AppSettings["name"]);
ChangeConfiguration();
Console.WriteLine(ConfigurationManager.AppSettings["name"]);
Console.ReadLine();
} private static void ChangeConfiguration()
{
//读取程序集的配置文件
string assemblyConfigFile = Assembly.GetEntryAssembly().Location; Configuration config = ConfigurationManager.OpenExeConfiguration(assemblyConfigFile);
//获取appSettings节点
AppSettingsSection appSettings = (AppSettingsSection)config.GetSection("appSettings"); //删除name,然后添加新值
appSettings.Settings.Remove("name");
appSettings.Settings.Add("name", "new"); //保存配置文件
config.Save();
}

清理,重新运行:

使用默认的不传递字符串的版本就可以打开当前配置文件了。

Configuration config = ConfigurationManager.OpenExeConfiguration(ConfigurationUserLevel.None);

如果要查看当前配置文件的完整路径可以使用AppDomain.CurrentDomain.SetupInformation.ConfigurationFile;

重新运行,结果如下:

在WCF程序中动态修改app.config配置文件

参考

https://blogs.msdn.microsoft.com/youssefm/2010/01/21/how-to-change-net-configuration-files-at-runtime-including-for-wcf/

http://developer.51cto.com/art/200908/146303.htm

http://www.codeproject.com/Articles/14744/Read-Write-App-Config-File-with-NET

上一篇:恢复MySQL备份时查看问题


下一篇:我可以从数据文件中找出哪个版本的MySQL?