我重写它有一个XmlReader传递的方法,我需要找到一个特定的元素,添加属性,然后创建一个新的XmlReader或只是取代现有的与修改的内容。 我使用C#4.0
使用的XElement(LINQ)我已经调查了,但我似乎无法操作一个现有的元素,并添加属性和价值。
我知道的XmlWriter具有WriteAttributeString这将是非常美妙的,但我又是不知道这一切是如何结合在一起的
我希望能够做这样的事情---这是伪代码!
public XmlReader DoSomethingWonderful(XmlReader reader)
{
Element element = reader.GetElement("Test");
element.SetAttribute("TestAttribute","This is a test");
reader.UpdateElement(element);
return reader;
}
的XmlReader /写入器顺序访问流。 您必须阅读的一端,过程你想怎么流,并写入另一端。 其优点是,你并不需要阅读整个事情到内存中,并建立一个DOM,这是你与任何基于的XmlDocument的方法得到的。
这个方法应该让你开始:
private static void PostProcess(Stream inStream, Stream outStream)
{
var settings = new XmlWriterSettings() { Indent = true, IndentChars = " " };
using (var reader = XmlReader.Create(inStream))
using (var writer = XmlWriter.Create(outStream, settings)) {
while (reader.Read()) {
switch (reader.NodeType) {
case XmlNodeType.Element:
writer.WriteStartElement(reader.Prefix, reader.Name, reader.NamespaceURI);
writer.WriteAttributes(reader, true);
//
// check if this is the node you want, inject attributes here.
//
if (reader.IsEmptyElement) {
writer.WriteEndElement();
}
break;
case XmlNodeType.Text:
writer.WriteString(reader.Value);
break;
case XmlNodeType.EndElement:
writer.WriteFullEndElement();
break;
case XmlNodeType.XmlDeclaration:
case XmlNodeType.ProcessingInstruction:
writer.WriteProcessingInstruction(reader.Name, reader.Value);
break;
case XmlNodeType.SignificantWhitespace:
writer.WriteWhitespace(reader.Value);
break;
}
}
}
}
这是不是很派生自己的XmlWriter一样干净,但我发现它更容易。
[编辑]
如何你会打开两个流一次一个例子可能是这样的:
using (FileStream readStream = new FileStream(@"c:\myFile.xml", FileMode.OpenOrCreate, FileAccess.Read, FileShare.Write)) {
using (FileStream writeStream = new FileStream(@"c:\myFile.xml", FileMode.OpenOrCreate, FileAccess.Write)) {
PostProcess(readStream, writeStream);
}
}
你不能简单的与做XmlReader
-至少,不是没有从读者阅读整个XML文档中,它把玩,然后创建一个新XmlReader
从结果。 这违背了很多使用的点XmlReader
虽然-即以流大文件的能力。
你可能从得到XmlReader
,转发最多方法调用现有的读者,但他们拦截酌情添加额外的属性等等...但我怀疑的代码将是非常非常复杂和脆弱。
我固定它使用以下胶带编码
public XmlReader FixUpReader(XmlReader reader)
{
reader.MoveToContent();
string xml = reader.ReadOuterXml();
string dslVersion = GetDSLVersion();
string Id = GetID();
string processedValue = string.Format("<ExampleElement dslVersion=\"{1}\" Id=\"{2}\" ", dslVersion, Id);
xml = xml.Replace("<ExampleElement ", processedValue);
MemoryStream ms = new MemoryStream(System.Text.Encoding.ASCII.GetBytes(xml));
XmlReaderSettings settings = new XmlReaderSettings();
XmlReader myReader = XmlReader.Create(ms);
myReader.MoveToContent();
return myReader;
}
我觉得脏了做这种方式,但它是工作....
我宁愿将XML加载到XmlDocument
对象和使用Attributes
集合修改数值并调用Save
方法来更新这个值。 下面的代码为我工作。
public static void WriteElementValue ( string sName, string element, string value)
{
try
{
var node = String.Format("//elements/element[@name='{0}']", sName);
var doc = new XmlDocument { PreserveWhitespace = true };
doc.Load(configurationFileName);
var nodeObject = doc.SelectSingleNode(node);
if (nodeObject == null)
throw new XmlException(String.Format("{0} path does not found any matching
node", node));
var elementObject = nodeObject[element];
if (elementObject != null)
{
elementObject.Attributes["value"].Value = value;
}
doc.Save(configurationFileName);
}
catch (Exception ex)
{
throw new ExitLevelException(ex, false);
}
}
当您使用的XmlWriter或XmlSerializer的的空白字符没有被正确保存,我也观察到,这可能是恼人有时
string newvalue = "10";
string presentvalue = "";
string newstr = "";
XmlReader xmlr = XmlReader.Create(new StringReader(str));
while (xmlr.Read())
{
if (xmlr.NodeType == XmlNodeType.Element)
{
if (xmlr.Name == "priority")
{
presentvalue = xmlr.ReadElementContentAsString();
newstr = str.Replace(presentvalue, newvalue);
}
}
}
//可中newstr写回文件...这是编辑的XML