我正在尝试为.NET4应用程序编写一个非常简单的自定义配置部分。我的目标是:
<configuration>
<configSections>
<section name="myServices" type="My.ConfigSection, My.Assembly" />
</configSections>
<myServices>
<add name="First" />
<add name="Second" />
</myServices>
</configuration>
然而,当我致电ConfigurationErrorsException
时,我不断收到ConfigurationManager.GetSection("myServices")
:'无法识别的元素'添加'。我已经盯着它看了一段时间,但还没弄明白我做错了什么。以下是我的代码。这是三个类:ConfigSection
,MyServiceSettingsCollection
和MyServiceSettings
。
首先是代表整个配置部分的类。它有一个类型为MyServiceSettingsCollection
的无名默认集合。 IsDefaultCollection
属性应该允许我从根元素“直接”添加到我的集合中。
public sealed class ConfigSection : ConfigurationSection
{
private static readonly ConfigurationProperty _propMyServices;
private static readonly ConfigurationPropertyCollection _properties;
public static ConfigSection Instance { get { return _instance; } }
static ConfigSection()
{
_propMyServices = new ConfigurationProperty(
null, typeof(MyServiceSettingsCollection), null,
ConfigurationPropertyOptions.IsDefaultCollection);
_properties = new ConfigurationPropertyCollection { _propMyServices };
}
[ConfigurationProperty("", IsDefaultCollection = true)]
public MyServiceSettingsCollection MyServices
{
get { return (MyServiceSettingsCollection) base[_propMyServices]; }
set { base[_propMyServices] = value; }
}
protected override ConfigurationPropertyCollection Properties
{ get { return _properties; } }
}
接下来,收集类本身。它的类型为AddRemoveClearMap
。
[ConfigurationCollection(typeof(MyServiceSettings),
CollectionType = ConfigurationElementCollectionType.AddRemoveClearMap)]
public sealed class MyServiceSettingsCollection : ConfigurationElementCollection
{
public MyServiceSettings this[int index]
{
get { return (MyServiceSettings) BaseGet(index); }
set
{
if (BaseGet(index) != null) { BaseRemoveAt(index); }
BaseAdd(index, value);
}
}
public new MyServiceSettings this[string key]
{
get { return (MyServiceSettings) BaseGet(key); }
}
protected override ConfigurationElement CreateNewElement()
{
return new MyServiceSettings();
}
protected override object GetElementKey(ConfigurationElement element)
{
return ((MyServiceSettings) element).Key;
}
}
最后是集合中元素的类。目前,这个类有一个属性,但稍后会有更多(这阻止我使用NameValueSectionHandler
)。
public class MyServiceSettings : ConfigurationElement
{
private static readonly ConfigurationProperty _propName;
private static readonly ConfigurationPropertyCollection properties;
static MyServiceSettings()
{
_propName = new ConfigurationProperty("name", typeof(string), null, null,
new StringValidator(1),
ConfigurationPropertyOptions.IsRequired |
ConfigurationPropertyOptions.IsKey);
properties = new ConfigurationPropertyCollection { _propName };
}
[ConfigurationProperty("name", DefaultValue = "",
Options = ConfigurationPropertyOptions.IsRequired |
ConfigurationPropertyOptions.IsKey)]
public string Name
{
get { return (string) base[_propKey]; }
set { base[_propKey] = value; }
}
protected override ConfigurationPropertyCollection Properties
{ get { return properties; } }
}
答案 0 :(得分:12)
好的,我发现了看似随意的修复。而不是:
[ConfigurationProperty("", IsDefaultCollection = true)]
public ProvisiorServiceSettingsCollection ProvisiorServices
{ ... }
你应该使用:
[ConfigurationProperty("", Options = ConfigurationPropertyOptions.IsDefaultCollection)]
public ProvisiorServiceSettingsCollection ProvisiorServices
{ ... }
不知道两者之间有什么区别。对我来说,它们看起来非常相似......或者至少,没有任何迹象表明为什么一个人比另一个更受欢迎。
答案 1 :(得分:3)
似乎你遗漏了与此相似的东西
[ConfigurationProperty("urls", IsDefaultCollection = false)]
[ConfigurationCollection(typeof(UrlsCollection),
AddItemName = "add",
ClearItemsName = "clear",
RemoveItemName = "remove")]
有关详细信息,请参阅http://msdn.microsoft.com/en-us/library/system.configuration.configurationcollectionattribute.aspx
答案 2 :(得分:3)
因为我花了很多时间在这上面,所以我想添加一个我在本次提交中实现的真实世界示例:https://github.com/rhythmagency/formulate/commit/4d2a95e1a82eb6b3500ab0869b8f8b15bd3deaa9
这是我的web.config的目标(我能够实现):
<?xml version="1.0" encoding="utf-8"?>
<configuration>
<configSections>
<sectionGroup name="formulateConfiguration">
<section name="templates" type="formulate.app.Configuration.TemplatesConfigSection, formulate.app" requirePermission="false"/>
</sectionGroup>
</configSections>
<formulateConfiguration>
<templates>
<template name="Responsive" path="~/Views/Formulate/Responsive.Bootstrap.Angular.cshtml" />
</templates>
</formulateConfiguration>
</configuration>
这是最高级别“模板”配置部分的类:
namespace formulate.app.Configuration
{
// Namespaces.
using System.Configuration;
/// <summary>
/// A configuration section for Formulate templates.
/// </summary>
public class TemplatesConfigSection : ConfigurationSection
{
#region Properties
/// <summary>
/// The templates in this configuration section.
/// </summary>
[ConfigurationProperty("", IsDefaultCollection = true)]
[ConfigurationCollection(typeof(TemplateCollection), AddItemName = "template")]
public TemplateCollection Templates
{
get
{
return base[""] as TemplateCollection;
}
}
#endregion
}
}
这是下一个级别,集合类:
namespace formulate.app.Configuration
{
// Namespaces.
using System.Configuration;
/// <summary>
/// A collection of templates from the configuration.
/// </summary>
[ConfigurationCollection(typeof(TemplateElement))]
public class TemplateCollection : ConfigurationElementCollection
{
#region Methods
/// <summary>
/// Creates a new template element.
/// </summary>
/// <returns>The template element.</returns>
protected override ConfigurationElement CreateNewElement()
{
return new TemplateElement();
}
/// <summary>
/// Gets the key for an element.
/// </summary>
/// <param name="element">The element.</param>
/// <returns>The key.</returns>
protected override object GetElementKey(ConfigurationElement element)
{
return (element as TemplateElement).Name;
}
#endregion
}
}
这是最深层次的课程(个别模板):
namespace formulate.app.Configuration
{
// Namespaces.
using System.Configuration;
/// <summary>
/// A "template" configuration element.
/// </summary>
public class TemplateElement : ConfigurationElement
{
#region Constants
private const string DefaultPath = "~/*Replace Me*.cshtml";
#endregion
#region Properties
/// <summary>
/// The name of the template.
/// </summary>
[ConfigurationProperty("name", IsRequired = true)]
public string Name
{
get
{
return base["name"] as string;
}
set
{
this["name"] = value;
}
}
/// <summary>
/// The path to this template.
/// </summary>
/// <remarks>
/// Should start with "~" and end with ".cshtml".
/// </remarks>
[ConfigurationProperty("path", IsRequired = true, DefaultValue = DefaultPath)]
[RegexStringValidator(@"^~.*\.[cC][sS][hH][tT][mM][lL]$")]
public string Path
{
get
{
var result = base["path"] as string;
return result == DefaultPath ? null : result;
}
set
{
this["path"] = value;
}
}
#endregion
}
}
对我来说,重要的一点是在ConfigurationPropertyAttribute
中设置空字符串并将IsDefaultCollection设置为true。顺便说一句,我把我的配置放在一个外观文件中,如下所示:
<?xml version="1.0" encoding="utf-8" ?>
<templates>
<template name="Responsive" path="~/Views/Formulate/Responsive.Bootstrap.Angular.cshtml" />
</templates>
在这种情况下,我的web.config看起来像这样:
<?xml version="1.0" encoding="utf-8"?>
<configuration>
<configSections>
<sectionGroup name="formulateConfiguration">
<section name="templates" type="formulate.app.Configuration.TemplatesConfigSection, formulate.app" requirePermission="false"/>
</sectionGroup>
</configSections>
<formulateConfiguration>
<templates configSource="config\Formulate\templates.config"/>
</formulateConfiguration>
</configuration>
想想我会提到有人试图将其添加到外部文件中(外部文件中的根级项与web.config中的外部元素相同,这有点不直观) )。