XMLDocument.Importnode 失败
XMLDocument.Importnode fails
我目前正在做一个项目,我必须在其中创建 XML 节点并将它们插入 XML 文件中的特定位置。
XML节点是一个包含大约 90 个单元格的行。
<Row ss:AutoFitHeight=\"0\">
<Cell ss:StyleID=\"s77\"><Data ss:Type=\"String\">ABC</Data></Cell>
<Cell><Data ss:Type=\"Number\">100</Data></Cell>
<Cell ss:StyleID=\"s77\"><Data ss:Type=\"String\">ABC</Data></Cell>
<Cell><Data ss:Type=\"String\" x:Ticked=\"1\">---</Data></Cell>
</Row>
我通过 Stringbuilder 创建文档并使用以下方法读取它们:
using (StringReader sr = new StringReader((sbuilder.ToString())))
using (XmlTextReader xtr = new XmlTextReader(sr) { Namespaces = false })
{
xdoc.Load(xtr);
}
我要添加节点的 XML 文件是一个 Excel 2003 Sheet,有 3 个选项卡,但我只需要第一个。我加载它:
XmlDocument xtemplate = new XmlDocument();
xtemplate.Load(file);
并尝试将我的节点导入模板 XML
XmlNode tempnode = xtemplate.ImportNode(xdoc.DocumentElement, true);
我在这里得到一个 XmlException
The ':' - characters Hex value 0x3A, shall not be included in a name.
我不知道这是从哪里来的。没有单元格或行获得自定义名称,我找不到包含除字母以外的任何内容的名称(名称为“#document”的文档除外)
我希望有人也遇到过这个异常,可以帮助我理解它,这样我就可以解决我的问题。
我希望你能理解我的问题是什么,如果没有我会尽力解释得更好。
我的英语有点生疏所以如果我屠杀它请怜悯。
更新:
它现在正在使用来自@dbc 的解决方案和一个小的添加:
string innerXml = @"<Row
xmlns=""urn:schemas-microsoft-com:office:spreadsheet""
xmlns:o=""urn:schemas-microsoft-com:office:office""
xmlns:x=""urn:schemas-microsoft-com:office:excel""
xmlns:ss=""urn:schemas-microsoft-com:office:spreadsheet""
xmlns:html=""http://www.w3.org/TR/REC-html40""
ss:AutoFitHeight=""0"">
<Cell ss:StyleID=""s77"">
<Data ss:Type=""String"">ABC</Data>
</Cell>
<Cell>
<Data ss:Type=""Number"">100</Data>
</Cell>
<Cell ss:StyleID=""s77"">
<Data ss:Type=""String"">ABC</Data>
</Cell>
<Cell>
<Data ss:Type=""String"" x:Ticked=""1"">---</Data>
</Cell>
</Row>";
var xdoc = new XmlDocument();
xdoc.LoadXml(innerXml); // Do not use new XmlTextReader(sr) { Namespaces = false })
XmlDocument xtemplate = new XmlDocument();
// Load xtemplate from file as before
XmlNode tempnode = xtemplate.ImportNode(xdoc.DocumentElement, true);
// Insert into Target Document
tempnode.ChildNodes[0].ParentNode.Attributes.RemoveAt(0); // Times 5 to delete all Namespacedefinitions
问题是,在您的 XML 片段中,您没有定义对应于命名空间前缀 ss
和 x
的 namespaces。命名空间前缀只是对给定元素范围内实际命名空间 table 的缩写查找。于是下面
<ss:Row xmlns:ss="http://somenamespace"/>
<xxxx:Row xmlns:xxxx="http://somenamespace"/>
<Row xmlns="http://somenamespace"/>
都定义相同的东西 -- 一个名为 {http://somenamespace}Row
.
的 XML 元素
当 ImportNode
将您的节点从一个文档复制到另一个文档时,它会尝试将名称空间前缀从旧文档映射到新文档以保留每个节点的实际名称空间——但是您的前缀丢失了它们的名称空间定义,因此无法执行映射并抛出异常。
要更正此问题,您必须将正确的命名空间定义添加到您的 XML 片段中,与 Excel 2003 XML 格式使用的命名空间定义一致。 Office 2003: XML Reference Schemas will tell you the correct namespaces. Having done so, you should use XmlDocument.LoadXml()
加载 XML 片段,然后来自 ImportNode
的异常将消失。
例如:
string innerXml = @"<Row
xmlns=""urn:schemas-microsoft-com:office:spreadsheet""
xmlns:o=""urn:schemas-microsoft-com:office:office""
xmlns:x=""urn:schemas-microsoft-com:office:excel""
xmlns:ss=""urn:schemas-microsoft-com:office:spreadsheet""
xmlns:html=""http://www.w3.org/TR/REC-html40""
ss:AutoFitHeight=""0"">
<Cell ss:StyleID=""s77"">
<Data ss:Type=""String"">ABC</Data>
</Cell>
<Cell>
<Data ss:Type=""Number"">100</Data>
</Cell>
<Cell ss:StyleID=""s77"">
<Data ss:Type=""String"">ABC</Data>
</Cell>
<Cell>
<Data ss:Type=""String"" x:Ticked=""1"">---</Data>
</Cell>
</Row>";
var xdoc = new XmlDocument();
xdoc.LoadXml(innerXml); // Do not use new XmlTextReader(sr) { Namespaces = false })
XmlDocument xtemplate = new XmlDocument();
// Load xtemplate from file as before
XmlNode tempnode = xtemplate.ImportNode(xdoc.DocumentElement, true); // No exception
我目前正在做一个项目,我必须在其中创建 XML 节点并将它们插入 XML 文件中的特定位置。
XML节点是一个包含大约 90 个单元格的行。
<Row ss:AutoFitHeight=\"0\">
<Cell ss:StyleID=\"s77\"><Data ss:Type=\"String\">ABC</Data></Cell>
<Cell><Data ss:Type=\"Number\">100</Data></Cell>
<Cell ss:StyleID=\"s77\"><Data ss:Type=\"String\">ABC</Data></Cell>
<Cell><Data ss:Type=\"String\" x:Ticked=\"1\">---</Data></Cell>
</Row>
我通过 Stringbuilder 创建文档并使用以下方法读取它们:
using (StringReader sr = new StringReader((sbuilder.ToString())))
using (XmlTextReader xtr = new XmlTextReader(sr) { Namespaces = false })
{
xdoc.Load(xtr);
}
我要添加节点的 XML 文件是一个 Excel 2003 Sheet,有 3 个选项卡,但我只需要第一个。我加载它:
XmlDocument xtemplate = new XmlDocument();
xtemplate.Load(file);
并尝试将我的节点导入模板 XML
XmlNode tempnode = xtemplate.ImportNode(xdoc.DocumentElement, true);
我在这里得到一个 XmlException
The ':' - characters Hex value 0x3A, shall not be included in a name.
我不知道这是从哪里来的。没有单元格或行获得自定义名称,我找不到包含除字母以外的任何内容的名称(名称为“#document”的文档除外)
我希望有人也遇到过这个异常,可以帮助我理解它,这样我就可以解决我的问题。
我希望你能理解我的问题是什么,如果没有我会尽力解释得更好。 我的英语有点生疏所以如果我屠杀它请怜悯。
更新:
它现在正在使用来自@dbc 的解决方案和一个小的添加:
string innerXml = @"<Row
xmlns=""urn:schemas-microsoft-com:office:spreadsheet""
xmlns:o=""urn:schemas-microsoft-com:office:office""
xmlns:x=""urn:schemas-microsoft-com:office:excel""
xmlns:ss=""urn:schemas-microsoft-com:office:spreadsheet""
xmlns:html=""http://www.w3.org/TR/REC-html40""
ss:AutoFitHeight=""0"">
<Cell ss:StyleID=""s77"">
<Data ss:Type=""String"">ABC</Data>
</Cell>
<Cell>
<Data ss:Type=""Number"">100</Data>
</Cell>
<Cell ss:StyleID=""s77"">
<Data ss:Type=""String"">ABC</Data>
</Cell>
<Cell>
<Data ss:Type=""String"" x:Ticked=""1"">---</Data>
</Cell>
</Row>";
var xdoc = new XmlDocument();
xdoc.LoadXml(innerXml); // Do not use new XmlTextReader(sr) { Namespaces = false })
XmlDocument xtemplate = new XmlDocument();
// Load xtemplate from file as before
XmlNode tempnode = xtemplate.ImportNode(xdoc.DocumentElement, true);
// Insert into Target Document
tempnode.ChildNodes[0].ParentNode.Attributes.RemoveAt(0); // Times 5 to delete all Namespacedefinitions
问题是,在您的 XML 片段中,您没有定义对应于命名空间前缀 ss
和 x
的 namespaces。命名空间前缀只是对给定元素范围内实际命名空间 table 的缩写查找。于是下面
<ss:Row xmlns:ss="http://somenamespace"/>
<xxxx:Row xmlns:xxxx="http://somenamespace"/>
<Row xmlns="http://somenamespace"/>
都定义相同的东西 -- 一个名为 {http://somenamespace}Row
.
当 ImportNode
将您的节点从一个文档复制到另一个文档时,它会尝试将名称空间前缀从旧文档映射到新文档以保留每个节点的实际名称空间——但是您的前缀丢失了它们的名称空间定义,因此无法执行映射并抛出异常。
要更正此问题,您必须将正确的命名空间定义添加到您的 XML 片段中,与 Excel 2003 XML 格式使用的命名空间定义一致。 Office 2003: XML Reference Schemas will tell you the correct namespaces. Having done so, you should use XmlDocument.LoadXml()
加载 XML 片段,然后来自 ImportNode
的异常将消失。
例如:
string innerXml = @"<Row
xmlns=""urn:schemas-microsoft-com:office:spreadsheet""
xmlns:o=""urn:schemas-microsoft-com:office:office""
xmlns:x=""urn:schemas-microsoft-com:office:excel""
xmlns:ss=""urn:schemas-microsoft-com:office:spreadsheet""
xmlns:html=""http://www.w3.org/TR/REC-html40""
ss:AutoFitHeight=""0"">
<Cell ss:StyleID=""s77"">
<Data ss:Type=""String"">ABC</Data>
</Cell>
<Cell>
<Data ss:Type=""Number"">100</Data>
</Cell>
<Cell ss:StyleID=""s77"">
<Data ss:Type=""String"">ABC</Data>
</Cell>
<Cell>
<Data ss:Type=""String"" x:Ticked=""1"">---</Data>
</Cell>
</Row>";
var xdoc = new XmlDocument();
xdoc.LoadXml(innerXml); // Do not use new XmlTextReader(sr) { Namespaces = false })
XmlDocument xtemplate = new XmlDocument();
// Load xtemplate from file as before
XmlNode tempnode = xtemplate.ImportNode(xdoc.DocumentElement, true); // No exception