您使用XML执行的所有操作都区分大小写,我知道。
然而,现在我发现自己处于这样一种情况,即如果我以某种方式使xml名称/属性识别不敏感,那么我正在编写的软件会产生更少的错误。不区分大小写的XPath将是一个神派。
在c#中有一个简单的方法/库吗?
答案 0 :(得分:14)
XMl文档可以分别命名为两个不同的元素:MyName
和myName
- 这些元素是不同的。将它们转换/处理为同名是一个可能产生严重后果的错误。
如果不是上述情况,那么这里有一个更精确的解决方案,使用XSLT将文档处理成只有小写元素名称和小写属性名称的文档:
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:variable name="vUpper" select=
"'ABCDEFGHIJKLMNOPQRSTUVWXYZ'"/>
<xsl:variable name="vLower" select=
"'abcdefghijklmnopqrstuvwxyz'"/>
<xsl:template match="node()|@*">
<xsl:copy>
<xsl:apply-templates select="node()|@*"/>
</xsl:copy>
</xsl:template>
<xsl:template match="*[name()=local-name()]" priority="2">
<xsl:element name="{translate(name(), $vUpper, $vLower)}"
namespace="{namespace-uri()}">
<xsl:apply-templates select="node()|@*"/>
</xsl:element>
</xsl:template>
<xsl:template match="*" priority="1">
<xsl:element name=
"{substring-before(name(), ':')}:{translate(local-name(), $vUpper, $vLower)}"
namespace="{namespace-uri()}">
<xsl:apply-templates select="node()|@*"/>
</xsl:element>
</xsl:template>
<xsl:template match="@*[name()=local-name()]" priority="2">
<xsl:attribute name="{translate(name(), $vUpper, $vLower)}"
namespace="{namespace-uri()}">
<xsl:value-of select="."/>
</xsl:attribute>
</xsl:template>
<xsl:template match="@*" priority="1">
<xsl:attribute name=
"{substring-before(name(), ':')}:{translate(local-name(), $vUpper, $vLower)}"
namespace="{namespace-uri()}">
<xsl:value-of select="."/>
</xsl:attribute>
</xsl:template>
</xsl:stylesheet>
将此转换应用于任何XML文档时,例如:
<authors xmlns:user="myNamespace">
<?ttt This is a PI ?>
<Author xmlns:user2="myNamespace2">
<Name idd="VH">Victor Hugo</Name>
<user2:Name idd="VH">Victor Hugo</user2:Name>
<Nationality xmlns:user3="myNamespace3">French</Nationality>
</Author>
<!-- This is a very long comment the purpose is
to test the default stylesheet for long comments-->
<Author Period="classical">
<Name>Sophocles</Name>
<Nationality>Greek</Nationality>
</Author>
<author>
<Name>Leo Tolstoy</Name>
<Nationality>Russian</Nationality>
</author>
<Author>
<Name>Alexander Pushkin</Name>
<Nationality>Russian</Nationality>
</Author>
<Author Period="classical">
<Name>Plato</Name>
<Nationality>Greek</Nationality>
</Author>
</authors>
生成了想要的正确结果(转换为小写的元素和属性名称):
<authors><?ttt This is a PI ?>
<author>
<name idd="VH">Victor Hugo</name>
<user2:name xmlns:user2="myNamespace2" idd="VH">Victor Hugo</user2:name>
<nationality>French</nationality>
</author><!-- This is a very long comment the purpose is
to test the default stylesheet for long comments-->
<author period="classical">
<name>Sophocles</name>
<nationality>Greek</nationality>
</author>
<author>
<name>Leo Tolstoy</name>
<nationality>Russian</nationality>
</author>
<author>
<name>Alexander Pushkin</name>
<nationality>Russian</nationality>
</author>
<author period="classical">
<name>Plato</name>
<nationality>Greek</nationality>
</author>
</authors>
将文档转换为所需的表单后,即可对转换后的文档执行任何所需的处理。
答案 1 :(得分:13)
您可以创建不区分大小写的方法(可用性扩展),例如:
public static class XDocumentExtensions
{
public static IEnumerable<XElement> ElementsCaseInsensitive(this XContainer source,
XName name)
{
return source.Elements()
.Where(e => e.Name.Namespace == name.Namespace
&& e.Name.LocalName.Equals(name.LocalName, StringComparison.OrdinalIgnoreCase));
}
}
答案 2 :(得分:7)
XML是文本。在加载到您正在使用的任何解析器之前,只需ToLower
。
只要您不必对模式进行验证并且不介意值全部为小写,这应该可以正常工作。
事实是任何XML解析器都区分大小写。如果不是,它将不是XML解析器。
答案 3 :(得分:0)
我首先将所有标签和属性名称转换为小写,通过使用SAX
解析保持值不变,即。与XmlTextReader
。
答案 4 :(得分:0)
我使用另一种解决方案。人们之所以想要这样做,是因为您也不想在类文件中的属性中重复该属性的名称。所以我要做的是向所有属性添加自定义属性:
[AttributeUsage(AttributeTargets.Property)]
public class UsePropertyNameToLowerAsXmlElementAttribute: XmlElementAttribute
{
public UsePropertyNameToLowerAsXmlElementAttribute([CallerMemberName] string propertyName = null)
: base(propertyName?.ToLower())
{
}
}
通过这种方式,XML序列化程序可以将小写属性映射到CamelCased类。
类上的属性仍然具有一个装饰器,该装饰器表示有所不同,但是您不必为每个属性都加上一个名称来进行标记:
public class Settings
{
[UsePropertyNameToLowerAsXmlElement]
public string VersionId { get; set; }
[UsePropertyNameToLowerAsXmlElement]
public int? ApplicationId { get; set; }
}