Как загрузить и объединить набор данных XML-документов - PullRequest
1 голос
/ 24 марта 2011

Я хотел бы использовать набор данных XML-документов и объединить их в один документ, содержащий только отдельные элементы.

Для иллюстрации у меня есть набор данных:

r, x
-- -------------------------------
1, <root><a>111</a></root>
2, <root><a>222</a><b>222</b></root>
3, <root><c>333</c></root>

приведет к:

<a>111</a><b>222</b><c>333</c>

Элемент <a> из r = 2 не объединяется, поскольку у нас уже есть элемент = <a> из r = 1. Мне нужно только объединить новые элементы, начиная с r = 1 и далее.

Я могу перебирать список, но мне сложно сравнивать и объединять. Приведенный ниже код не может идентифицировать <a>222</a> как дубликат. Возможно ли сравнение значений элементов?

 using (SqlDataReader dsReader = cmd.ExecuteReader())
            {
                XDocument baseDoc = new XDocument();
                XDocument childDoc = new XDocument();

                while (dsReader.Read())
                {
                    // this is the base doc, merge forward from here
                    if (dsReader["r"].ToString() == "1")
                    {
                        baseDoc = XDocument.Parse(dsReader["x"].ToString());
                        SqlContext.Pipe.Send("start:" + baseDoc.ToString());

                    }
                    // this is a child doc, do merge operation
                    else
                    {
                        childDoc = XDocument.Parse(dsReader["x"].ToString());

                        // find elements only present in child
                        var childOnly = (childDoc.Descendants("root").Elements()).Except(baseDoc.Descendants("root").Elements());
                        foreach (var e in childOnly)
                        {
                            baseDoc.Root.Add(e);
                        }
                    }
                }
            }

1 Ответ

1 голос
/ 24 марта 2011

Я немного запутался по поводу использования baseDoc и childDoc в вашем коде. Надеюсь я правильно понял ваш вопрос. Вот мое предложение:

        using (SqlDataReader dsReader = cmd.ExecuteReader())
        {
            XElement result = new XElement("root");
            while (dsReader.Read())
            {
                // Read source
                XDocument srcDoc = XDocument.Parse(dsReader["x"].ToString());

                // Construct result element
                foreach (XElement baseElement in srcDoc.Descendants("root").Elements())
                    if (result.Element(baseElement.Name) == null)   // skip already added nodes
                        result.Add(new XElement(baseElement.Name, baseElement.Value));

            }
            // Construct result string from sub-elements (to avoid "<root>..</root>" in output)
            string str = "";
            foreach (XElement element in result.Elements())
                str += element.ToString();

            // send the result
            SqlContext.Pipe.Send("start:" + str);
        }

Обратите внимание, что мой код игнорирует r-нумерацию. Я использую порядок, поскольку он прибывает из читателя данных SQL. Если строки не отсортированы по "r", перед моим кодом требуется дополнительная сортировка.

...