如何编写XQuery访问存储在单独的SQL Server表中的XML?

时间:2013-02-21 14:28:16

标签: sql-server xquery xquery-sql

我创建了一个包含两个表的新数据库来测试SQLServer 2008 XML功能。

如果我成功获得所需的行为,我将不得不创建大约50个表,然后为新项目导入大量XML文件。

我的测试环境是这样创建的:

create table Employees(
    idEmployeeFeed bigint primary key IDENTITY(1,1),
    EmployeeFeed xml
)
go

create table GeoCountries(
    CountriesFeed xml
)
go

然后我在Employees表中加载了大约1000个xml文件,在GeoCountries表中加载了1个文件。 GeoCountries文件包含249个国家/地区的纬度/长度质心坐标以及国家/地区名称和ISO 3字符国家/地区代码。

每位员工都有一个国家/地区代码。我已经在另一个产品中准备好了XQuery,现在我需要根据客户端要求迁移到SQL Server。

使用XQuery获取的两个表中的示例数据:

select EmployeeFeed.query('//employee') as employee
from Employees
/* output:
    <employee empID="1111" >
      <displayName>John</displayName>
      <country code="USA" />
    </employee>
    <employee empID="2222" >
      <displayName>Mario</displayName>
      <country code="ITA" />
    </employee>
    ...
*/

select EmployeeFeed.query('//employee/country') as employee
from Employees
/* output:
    <country code="USA" />
    <country code="ITA" />
    ...
*/

select CountriesFeed.query('//country')
from GeoCountries
/* output:
    <country>
      <ISO3166A3>USA</ISO3166A3>
      <ISOen_name>United States</ISOen_name>
      <latitude>38.000</latitude>
      <longitude>-97.000</longitude>
    </country>
    <country>
      <ISO3166A3>ITA</ISO3166A3>
      <ISOen_name>Italy</ISOen_name>
      <latitude>42.833</latitude>
      <longitude>12.833</longitude>
    </country>
    ...
*/

select CountriesFeed.query('//country/ISO3166A3')
from GeoCountries
/* output:
    <ISO3166A3>USA</ISO3166A3>
    <ISO3166A3>ITA</ISO3166A3>
    ...
*/

这是我试图运行的查询:

select EmployeeFeed.query(N'
  let $ccc := //country
  let $ttt := //employee
  for $t in $ttt  
  return
        <geoEmp
        empCode="{ $t/@empID }" 
        countryCode="{ $t/country/@code }" 
        latit="{ $ccc[ISO3166A3 = $t/country/@code]/latitude/text() }" 
        longit="{ $ccc[ISO3166A3 = $t/country/@code]/longitude/text() }"
        />
') as GeoEmployees
from Employees
/* output:
    <geoEmp empCode="1111" countryCode="USA" latit="" longit="" />
    <geoEmp empCode="2222" countryCode="ITA" latit="" longit="" />
    ...
*/

正如您所看到的,国家代码+ lat / long预先加载到变量$ ccc中以将其用作查找表,但是,作为另一个SQL表(GeoCountries)中的数据,它在当前不能找到与SQL表Employees绑定的XQuery上下文。

有没有办法运行XQuery访问存储在不同SQL表中的XML? 如果我迁移,我将有100-200个类似的情况需要管理,我需要为这个问题找到一个有效的解决方案。

1 个答案:

答案 0 :(得分:1)

这个怎么样:

select e.empID as '@empCode', c.code as '@countryCode', c.lat as '@latit', c.long as '@longit' from (
    select e.emp.value('(@empID)[1]', 'int') as empID, e.emp.value('(country/@code)[1]', 'varchar(32)') as code
    from Employees
    cross apply EmployeeFeed.nodes('//employee') e(emp)
) e
inner join (
    select c.country.value('(ISO3166A3)[1]', 'varchar(32)') as code, c.country.value('(latitude)[1]', 'varchar(32)') as lat, c.country.value('(longitude)[1]', 'varchar(32)') as long
    from GeoCountries
    cross apply CountriesFeed.nodes('//country') c(country)
) c on e.code = c.code
for xml path('geoEmp'), type