如何一次操作一个XML文档的一个父元素

本文关键字:一个 元素 文档 XML 操作 何一次 | 更新日期: 2023-09-27 18:20:19

我正试图从在线购物车中获取一个包含多个订单的XML文件,对其进行解析,并使用C#和Visual Studio 2008将每个订单的值输出为自己的文本文件(而不是XML)。在这一点上,我尝试了各种方法,但都没有成功。我的最后一次尝试包括一个绑定到XMLNodeList的foreach语句,并试图通过使用XMLReader将每个元素值写入字符串来执行名为"Order"的每个节点的翻译。foreach似乎不适用于当前配置。

有更好的方法吗?还是我需要继续使用前臂?我们非常感谢所有的想法。

 class Class1
  {
     public static void Main()
      {
        StringBuilder orderid = new StringBuilder();
        StringBuilder ordernumber = new StringBuilder();
        StringBuilder name = new StringBuilder();
        StringBuilder staddress = new StringBuilder();
        StringBuilder city = new StringBuilder();
        StringBuilder state = new StringBuilder();
        StringBuilder zip = new StringBuilder();
        StringBuilder country = new StringBuilder();
        StringBuilder email = new StringBuilder();
        StringBuilder partnumber = new StringBuilder();
        StringBuilder quantity = new StringBuilder();
            XmlDocument doc = new XmlDocument();
            doc.Load(@"C:'onlinesales'neworders.xml");
            XmlNode root = doc.DocumentElement;
            XmlNodeList nodeList = root.SelectNodes("Order");
            foreach (XmlNode order in nodeList)
            {
                using (XmlReader reader = XmlReader.Create("Order"))
                {
                    reader.ReadToFollowing("OrderNumber");
                    ordernumber.Append(reader.ReadElementContentAsString());
                    reader.ReadToFollowing("OrderGUID");
                    orderid.Append(reader.ReadElementContentAsString());
                    reader.ReadToFollowing("FirstName");
                    name.Append(reader.ReadElementContentAsString());
                    reader.ReadToFollowing("Email");
                    email.Append(reader.ReadElementContentAsString());
                    reader.ReadToFollowing("BillingAddress1");
                    staddress.Append(reader.ReadElementContentAsString());
                    reader.ReadToFollowing("BillingCity");
                    city.Append(reader.ReadElementContentAsString());
                    reader.ReadToFollowing("BillingState");
                    state.Append(reader.ReadElementContentAsString());
                    reader.ReadToFollowing("BillingZip");
                    zip.Append(reader.ReadElementContentAsString());
                    reader.ReadToFollowing("BillingCountry");
                    country.Append(reader.ReadElementContentAsString());
                    reader.ReadToFollowing("Quantity");
                    quantity.Append(reader.ReadElementContentAsString());
                    reader.ReadToFollowing("OrderedProductManufacturerPartNumber");
                    partnumber.Append(reader.ReadElementContentAsString());
                }
            using (StreamWriter fileout =
                new StreamWriter("W:" + DateTime.Now.ToString("yyyyy-MM-dd_hh-mm-ss-ff") + ".txt", false, Encoding.ASCII))
            {
                fileout.WriteLine("ISA*00*          *00*          *ZZ*daisywebstore  *12*5016361200     *" + DateTime.Now.ToString("yyMMdd") + "*1559*U*00400*000001649*0*P>~");
                fileout.WriteLine("GS*PO*daisywebstore*5016361200*" + DateTime.Now.ToString("yyyyMMdd") + "*" + DateTime.Now.ToString("HHmm") + "*1649*X*004010~");
                fileout.WriteLine("ST*850*13~");
                fileout.WriteLine("BEG*00*SA*08272226001*" + DateTime.Now.ToString("yyyyMMdd") + "~");
                fileout.WriteLine("REF*DP*089~");
                fileout.WriteLine("DTM*002*20120104~");
                fileout.WriteLine("N1*ST*" + name + "~");
                fileout.WriteLine("N3*" + staddress + "~");
                fileout.WriteLine("N4*" + city + "*" + state + "*" + zip + "~");
                fileout.WriteLine("N1*RE**92*00103653341~");
                fileout.WriteLine("PO1*1*6*EA*33.28*TE*IN*985880-542~");
                fileout.WriteLine("PID*F*****CO2 BB PISTOL     $ 5693~");
                fileout.WriteLine("PO4*3*1*EA~");
                fileout.WriteLine("CT*1~");
                fileout.WriteLine("AMT*1*199.68~");
                fileout.WriteLine("SE*16*13~");
            }
                    }
                //File.Delete(@"C:'onlinesales'neworders.xml");
}
}

}

如何一次操作一个XML文档的一个父元素

您可以让它变得超级简单。面向对象编程是可行的。

假设您创建了一个类Order,它接受一个XElement。我给你写几行。

public class Order
{
    XElement self;
    public Order(XElement order)
    {
        self = order;
    }
    public XElement Element { get { return self; } }
    public string OrderNumber 
    { 
        // if your xml looks like <Order OrderNumber="somenumber" />
        get { return (string)(self.Attribute("OrderNumber") ?? (object)"some default value/null"); } 
        // but if it looks like: <Order><OrderNumber>somenumber</OrderNumber></Order>
        // get { return (string)(self.Element("OrderNumber") ?? (object)"some default value/null"); }
    }
}

您必须自己填写订单的其余属性。对于每个Order值,创建一个类似于我上面创建的OrderNumber的属性。每个Order值都有属性,这使得访问数据变得非常简单。

所以对于你的主要代码,你会有:

XElement file = XElement.Load(@"C:'onlinesales'neworders.xml");
Order[] orders = file.Elements("Order").Select(e => new Order(e)).ToArray();

现在您已经将所有订单都作为单独的订单对象,在输出到文件时从属性列表中读取数据。现在不需要将值存储在StringBuilder中,因为值在Order对象中。

foreach(Order order in orders)
{
   // write order.OrderNumber  etc. / do whatever you want with the orders.
}
Chuck有一个很好的方法。有时,您不希望在正在加载的模型对象中保留对底层xml数据结构的引用。在这种情况下,我经常使用这样的模式:
public interface IXmlReadable
{
    void Clear();
    void Read(XPathNavigator xmlNav);
}
public class ModelBase : IXmlReadable
{
    public void Clear()
    {
        DoClear();
    }
    public void Read(XPathNavigator xmlNav)
    {
        DoRead(xmlNav);
    }

    protected virtual void DoClear()
    {
        throw new NotImplementedException();
    }
    protected virtual void DoRead(XPathNavigator xmlNav)
    {
        throw new NotImplementedException();
    }
}

读取可以过载以接受XmlDocumentXmlNodeXElement

现在您可以实现特定的模型。

public sealed class Order : ModelBase
{
    public Order() { }
    public string OrderNumber { get; private set; }
    protected override void DoClear()
    {
        OrderNumber = string.Empty;
    }
    protected override void DoRead(XPathNavigator xmlNav)
    {
        DoClear();
        XPathNavigator node;
        node = xmlNav.SelectSingleNode("OrderNumber");
        if (node != null)
            OrderNumber = node.InnerXml;
        // implement other properties here
    }
}

使用XPathDocument和XPathNavigator方法,您可以执行以下操作:

XPathDocument xml = new XPathDocument(@"C:'onlinesales'neworders.xml");
xmlNav = xml.CreateNavigator();
XPathNodeIterator iterator = xmlNav.Select("Order");
while (iterator.MoveNext())
{
    Order order = new Order();
    order.Read(iterator.Current);
    // do something with the Order - add to list or process
}