如何从XML *向SQL Server DATE字段*中插入NULL
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了如何从XML *向SQL Server DATE字段*中插入NULL相关的知识,希望对你有一定的参考价值。
我有一些XML,我试图使用他们的XML datatype functions插入到Microsoft SQL Server数据库中。
其中一个表字段是可以为空的DATE
列。如果节点丢失,那么它被插入为NULL
,这很棒。但是,如果节点存在但在运行XPath查询时为空<LastDay/>
,则它会将空节点中的值解释为empty string ''
instead of NULL
。因此,在查看表结果时,默认情况下会将日期转换为1900-01-01。
我想将空节点也插入为NULL
而不是默认的空字符串''
或1900-01-01。我怎样才能让它插入NULL
呢?
CREATE TABLE myxml
(
"id" INT,
"name" NVARCHAR(100),
"company" NVARCHAR(100),
"lastday" DATE
);
DECLARE @xml XML =
'<?xml version="1.0" encoding="UTF-8"?>
<Data xmlns="http://example.com" xmlns:dmd="http://example.com/data-metadata">
<Company dmd:name="Adventure Works Ltd.">
<Employee id="1">
<Name>John Doe</Name>
<LastDay>2016-08-01</LastDay>
</Employee>
<Employee id="2">
<Name>Jane Doe</Name>
</Employee>
</Company>
<Company dmd:name="StackUnderflow">
<Employee id="3">
<Name>Jeff Puckett</Name>
<LastDay/>
</Employee>
<Employee id="4">
<Name>Ill Gates</Name>
</Employee>
</Company>
</Data>';
WITH XMLNAMESPACES (DEFAULT 'http://example.com', 'http://example.com/data-metadata' as dmd)
INSERT INTO myxml (id,name,company,lastday)
SELECT
t.c.value('@id', 'INT' ),
t.c.value('Name[1]', 'VARCHAR(100)' ),
t.c.value('../@dmd:name','VARCHAR(100)' ),
t.c.value('LastDay[1]', 'DATE' )
FROM @xml.nodes('/Data/Company/Employee') t(c)
这会产生:
id name company lastday
------------------------------------------------
1 John Doe Adventure Works Ltd. 2016-08-01
2 Jane Doe Adventure Works Ltd. NULL
3 Jeff Puckett StackUnderflow 1900-01-01
4 Ill Gates StackUnderflow NULL
我想要实现:
id name company lastday
------------------------------------------------
1 John Doe Adventure Works Ltd. 2016-08-01
2 Jane Doe Adventure Works Ltd. NULL
3 Jeff Puckett StackUnderflow NULL
4 Ill Gates StackUnderflow NULL
您必须使用NULLIF
函数来避免从XML选择中弹出默认值。
如果两个指定的表达式相等,则返回null值。
您的查询将更改如下:
SELECT
t.c.value('@id', 'INT' ),
t.c.value('Name[1]','VARCHAR(100)' ),
t.c.value('../@dmd:name', 'VARCHAR(100)' ),
NULLIF(t.c.value('LastDay[1]', 'DATE' ),'')
FROM @xml.nodes('/Data/Company/Employee') t(c)
有关NULLIF
的更多信息,请查看this MSDN page。
除了techspider的非常好的答案,我想展示另一种方法:
在公司上执行.nodes()
,在Employee上执行CROSS APPLY .nodes()
,可以实现更清晰的XPath导航,并避免../@dmd.name
使用的向后导航。在你的情况下,这可能只是为了信息,但很好的考虑:如果有一个没有任何员工的公司,你会跳过整个公司,否则...(我的代码将跳过由于CROSS APPLY
,但你可以使用OUTER APPLY
)。
对于你的实际问题:使用内部cast as xs:date
将在XQuery中执行逻辑并且应该更快......
WITH XMLNAMESPACES (DEFAULT 'http://example.com', 'http://example.com/data-metadata' as dmd)
INSERT INTO myxml (id,name,company,lastday)
SELECT
e.value('@id', 'INT' ),
e.value('Name[1]', 'VARCHAR(100)' ),
c.value('@dmd:name', 'VARCHAR(100)' ),
e.value('let $x:=LastDay[1] return $x cast as xs:date?','DATE' )
FROM @xml.nodes('/Data/Company') AS A(c)
CROSS APPLY c.nodes('Employee') AS B(e)
以上是关于如何从XML *向SQL Server DATE字段*中插入NULL的主要内容,如果未能解决你的问题,请参考以下文章
如何使用 Spark 从 XML 复制到 SQL Server
如何从 SQL Server 中的 XML 元素获取特定属性