0

我试图了解如何组合 LINQ-to-XML 和 LINQ-to-SQL 查询并执行连接。

具体来说,我有一个包含城市、县和州信息的 SQL 表,我可以使用 LINQ to SQL 对其进行查询,但在同一个查询中,我想加入具有相同州和/或县,并生成一个 XML 作为输出的一部分。

这大概是我的桌子的样子:

╔═════╦══════════════╦════════════════╦═══════╗
║ IDX ║     CITY     ║   COUNTY       ║ STATE ║ 
╠═════╬══════════════╬════════════════╬═══════╣
║  1  ║ YAKUTAT      ║ ALEUTIANS EAST ║ AK    ║
║  2  ║ city-1       ║ ALEUTIANS EAST ║ AK    ║
║  3  ║ city-2       ║ ALEUTIANS EAST ║ AK    ║
║  4  ║ city-3       ║ ALEUTIANS WEST ║ AK    ║
║  5  ║ city-4       ║ ALEUTIANS WEST ║ AK    ║
║  6  ║ city-5       ║ ALEUTIANS WEST ║ AK    ║
║  7  ║ xyz          ║ ANCHORAGE      ║ AK    ║
║  8  ║ abc          ║ BETHEL         ║ AK    ║
║  9  ║ lmnop        ║ WYOMING        ║ NY    ║
║  10 ║ pqrst        ║ WARSAW         ║ NY    ║
║  11 ║ defg         ║ WARSAW         ║ NY    ║
╚═════╩══════════════╩════════════════╩═══════╝

这就是我希望我的 XML 输出的样子。我希望加入将具有相同县的所有城市分组为同一县节点下的节点,然后将一个州的所有县分组为该州节点的子节点。

<State>AK</State>
  <County>ALEUTIANS EAST</County>
    <City>YAKUTAT</City>
    <City>city-1</City>
    <City>city-2</City>
  <County>ALEUTIANS WEST</County>
    <City>city-3</City>
    <City>city-4</City>
    <City>city-5</City>   
  <County>ANCHORAGE</County>
    <City>xyz</City>
  <County>BETHEL</County>
    <City>abc</City>
<State>NY</State>
  <County>WYOMING</County>
    <City>lmnop</City>
  <County>WARSAW</County>
    <City>pqrst</City>
    <City>defg</City>

我确实有这部分工作,我能够成功地从我的数据库中选择行,并且我能够将输出编写为 XML,但我无法将城市和县数据嵌套为州的子节点,我错过了加入部分,目前尚不清楚如何做到这一点。

XDocument xDoc = new XDocument(new XElement("States",
   (from states in state.Database 
    select new XElement(new XElement("State",states.State),
           new XElelment("County",states.County),
           new XElelment("City",state.City))));

xDoc.Save("C:\\states.xml")

这就是我得到的输出。如您所见,我在我的数据库中获得了所有城市、州和县的完整列表,没有层次结构。

<State>AK</State>
<County>ALEUTIANS EAST</County>
<City>YAKUTAT</City>
<State>AK</State>
<County>ALEUTIANS EAST</County>
<City>city-1</City>
<State>AK</State>
<County>ALEUTIANS EAST</County>
<City>city-2</City>
<State>AK</State>
<County>ALEUTIANS WEST</County>
<City>city-3</City>
<State>AK</State>
<County>ALEUTIANS WEST</County>
<City>city-4</City>
....and so on..
4

1 回答 1

0

您确实有几个单独的问题,首先是您无法生成分层 XML 文件(假设我在编辑您的问题之前了解您的初衷)。事实上,您的示例 XML 甚至不是分层的,因为您在将预期的子标签嵌套在它下面之前关闭标签。像你正在做的缩进不是创建任何类型的子节点,只是让它看起来像。

这就是我认为您希望您的 XML 看起来像的样子:

<State>AK
  <County>ALEUTIANS EAST
    <City>YAKUTAT</city>
    <City>city-1</city>
  </County>
</State>

但是将字符串等简单类型与同一节点中的其他节点等复杂类型混合起来可能会非常混乱。我可以建议修改结构以使用属性,从而消除一些令人困惑的层次结构。像这样的结构更普遍可以接受:

<State name = "AK">
  <County name = "ALEUTIANS EAST">
    <City>YAKUTAT</city>
    <City>city-1</city>
  </County>
</State>

生成嵌套 XML 只需将XElement对象彼此排列在一起

XElement stateElement = new XElement("State", new XAttribute("name", "AK"), 
                            new XElement("County", new XAttribute("name", "ALEUTIANS EAST"), 
                                new XElement("City", "YAKUTAT"), 
                                new XElement("City", "city-1")));

将生成一个具有我上面建议的结构的 XML。

您缺少的第二部分是联接。由于所有内容都在同一个表中,因此您实际上是在寻找组查询,因为您想将子元素组合在一个共同的父元素下。

我不是对 LINQ-to-SQL 的期望,而且不同风格之间可能存在字幕差异这一事实,我不能 100% 确定语法是否完全正确,但希望如果它关闭,它会仍然让你指向正确的方向。

请注意,仅仅因为您正在创建一个 XML 文件,这不是 LINQ-to-XML,LINQ 是关于查询源而不是输出,因为您正在查询数据库,所以这是 LINQ-to-SQL

var query = from states in state.Database 
            group states by new { states.State, states.County } into countyGroup
            group countyGroup by countyGroup.Key.State into stateGroup
            select new XElement("State", new XAttribute("name", stateGroup.Key),
                       stateGroup.Select(st => new XElement("County", new XAttribute("name", st.Key.County),
                       st.Select(city => new XElement("City", city.City)))));

var xdoc = new XDocument(new XDeclaration("1.0", "UTF-8", null),
                         new XElement("States", query.ToArray()));

第一个 group 语句将根据共同的县和州将所有数据库行分组(必须包括州,就好像您将其排除在 Orange, FL 和 Orange, CA 内的城市一样。第二组语句然后将您的数据库行进一步分组到常见的州。您最终得到一个分组(称为stateGroup由同一县的城市集合和同一城市中的县组成的集合。

将分组分解为适当的 XML 节点的 select 语句变得非常混乱,但它有效地写出了每个分组,以便到达我认为您想要去的地方。

(当您在问题中创建您的问题时,您也错过了 XML 声明XDocument,所以我在此处添加了该声明)。

于 2013-03-08T10:21:29.257 回答