SQL Server:如何使用节点的字段导入XML

弗朗切斯科·曼托瓦尼(Francesco Mantovani)

我正在遵循该指南该指南教我如何将XML文件导入SQL Server。

我的XML如下所示:

<?xml version='1.0' encoding='UTF-8'?>
<osm version="0.6" generator="Osmosis 0.46">
  <bounds minlon="-180.00000" minlat="-90.00000" maxlon="180.00000" maxlat="90.00000" origin="http://www.openstreetmap.org/api/0.6"/>
  <node id="1014954" version="5" timestamp="2016-01-09T23:33:09Z" uid="1894634" user="Wikilux" changeset="36472980" lat="46.4928487" lon="7.5628558">
    <tag k="url" v="www.cine-rex.ch"/>
    <tag k="name" v="Ciné Rex"/>
    <tag k="amenity" v="cinema"/>
    <tag k="website" v="http://www.cine-rex.ch/kino.shtml"/>
    <tag k="addr:street" v="Landstrasse"/>
    <tag k="addr:housenumber" v="18"/>
  </node>
  <node id="20823872" version="7" timestamp="2017-09-12T15:19:00Z" uid="364" user="Edward" changeset="51976823" lat="52.2062941" lon="0.1346864">
    <tag k="name" v="Vue Cambridge"/>
    <tag k="screen" v="8"/>
    <tag k="amenity" v="cinema"/>
    <tag k="operator" v="Vue Cinemas"/>
    <tag k="wikidata" v="Q39197413"/>
    <tag k="cinema:3D" v="yes"/>
    <tag k="addr:street" v="The Grafton Centre"/>
    <tag k="addr:postcode" v="CB1 1PS"/>
  </node>
  <node id="20922159" version="12" timestamp="2017-09-12T15:19:00Z" uid="364" user="Edward" changeset="51976823" lat="52.2028721" lon="0.1234498">
    <tag k="name" v="Arts Picturehouse"/>
    <tag k="screen" v="3"/>
    <tag k="amenity" v="cinema"/>
    <tag k="operator" v="City Screen Limited"/>
    <tag k="wikidata" v="Q39197264"/>
    <tag k="addr:city" v="Cambridge"/>
    <tag k="wheelchair" v="no"/>
    <tag k="addr:street" v="St Andrew&apos;s Street"/>
    <tag k="addr:country" v="GB"/>
    <tag k="addr:postcode" v="CB2 3AR"/>
    <tag k="addr:housenumber" v="38-39"/>
  </node>
</osm>

所以我首先以这种方式导入XML:

CREATE DATABASE OPENXMLTesting
GO
USE OPENXMLTesting
GO
CREATE TABLE XMLwithOpenXML
(
Id INT IDENTITY PRIMARY KEY,
XMLData XML,
LoadedDateTime DATETIME
)
INSERT INTO XMLwithOpenXML(XMLData, LoadedDateTime)
SELECT CONVERT(XML, BulkColumn) AS BulkColumn, GETDATE() 
FROM OPENROWSET(BULK 'C:\Users\franc\Desktop\cinema.osm', SINGLE_BLOB) AS x;

SELECT * FROM XMLwithOpenXML

然后导入idlatlong从每个<node>

USE OPENXMLTesting
GO
DECLARE @XML AS XML, @hDoc AS INT, @SQL NVARCHAR (MAX)
SELECT @XML = XMLData FROM XMLwithOpenXML
EXEC sp_xml_preparedocument @hDoc OUTPUT, @XML
SELECT id, lat, lon
FROM OPENXML(@hDoc, 'osm/node')
WITH 
(
id [varchar](50) '@id',
lat [varchar](100) '@lat',
lon [varchar](100) '@lon'
)

EXEC sp_xml_removedocument @hDoc
GO

完美,有效!

但是现在我被卡住了。为了使用name创建更多2列,我需要做website什么?

在此处输入图片说明

因为<tag>总是相同的,所以我需要v=代表什么是k=

marc_s

可以通过SQL Server中的本机XQuery支持来执行此操作,而不必诉诸于OPENXML因内存泄漏和其他问题而闻名的旧式调用。

只需使用以下代码即可:

DECLARE @XML AS XML

SELECT @XML = XMLData FROM XMLwithOpenXML

SELECT
    id = xc.value('@id', 'int'),
    lon = xc.value('@lon', 'decimal(20,8)'),
    lat = xc.value('@lat', 'decimal(20,8)'),
    name = xc.value('(tag[@k="name"]/@v)[1]', 'varchar(50)'),
    website = xc.value('(tag[@k="website"]/@v)[1]', 'varchar(50)')
FROM
    @XML.nodes('/osm/node') AS XT(XC)

并且我还建议始终使用最合适的数据类型-在这里id看起来像INT-照此使用-和latandlon显然是DECIMAL值;不要将所有内容都转换为字符串,因为您太懒了以至于无法确定要使用什么!

本文收集自互联网,转载请注明来源。

如有侵权,请联系 [email protected] 删除。

编辑于
0

我来说两句

0 条评论
登录 后参与评论

相关文章