Sql server 从子节点openxml sql返回相同的节点值
我有以下XML:-Sql server 从子节点openxml sql返回相同的节点值,sql-server,stored-procedures,sql-server-2012,sql-server-openxml,Sql Server,Stored Procedures,Sql Server 2012,Sql Server Openxml,我有以下XML:- <XML> <ProductDetail> <ProductId>1</ProductId> <PropertyDetail> <PropertyKey>Size</PropertyKey> <PropertyValue>XXL</PropertyValue> <PropertyKey>ProdTaxType&
<XML>
<ProductDetail>
<ProductId>1</ProductId>
<PropertyDetail>
<PropertyKey>Size</PropertyKey>
<PropertyValue>XXL</PropertyValue>
<PropertyKey>ProdTaxType</PropertyKey>
<PropertyValue>5%</PropertyValue>
<PropertyKey>Incl/Excl</PropertyKey>
<PropertyValue>True</PropertyValue>
<PropertyKey>Fit</PropertyKey>
<PropertyValue>SLIM F/S</PropertyValue>
</PropertyDetail>
</ProductDetail>
</XML>
我需要使用openxml获取所有PropertyDetail。我在MS SQL 2012存储过程中的小查询如下:
ProductId PropertyKey PropertyValue
1 Size XXL
SELECT XML.ProductId, XML.PropertyKey, XML.PropertyValue FROM
OPENXML (@hDoc, '/XML/*', 2) WITH (
ProductId INT 'ProductId',
PropertyKey VARCHAR(200) 'PropertyDetail/PropertyKey',
PropertyValue VARCHAR(200) 'PropertyDetail/PropertyValue'
) XML
ProductId PropertyKey PropertyValue
1 Size XXL
1 ProdTaxType 5%
1 Incl/Excl True
1 Fit SLIM F/S
最终结果如下所示:
ProductId PropertyKey PropertyValue
1 Size XXL
SELECT XML.ProductId, XML.PropertyKey, XML.PropertyValue FROM
OPENXML (@hDoc, '/XML/*', 2) WITH (
ProductId INT 'ProductId',
PropertyKey VARCHAR(200) 'PropertyDetail/PropertyKey',
PropertyValue VARCHAR(200) 'PropertyDetail/PropertyValue'
) XML
ProductId PropertyKey PropertyValue
1 Size XXL
1 ProdTaxType 5%
1 Incl/Excl True
1 Fit SLIM F/S
我会尝试以下解决方案:
DECLARE @x XML = N'<XML>
<ProductDetail>
<ProductId>1</ProductId>
<PropertyDetail>
<PropertyKey>Size</PropertyKey>
<PropertyValue>XXL</PropertyValue>
<PropertyKey>ProdTaxType</PropertyKey>
<PropertyValue>5%</PropertyValue>
<PropertyKey>Incl/Excl</PropertyKey>
<PropertyValue>True</PropertyValue>
<PropertyKey>Fit</PropertyKey>
<PropertyValue>SLIM F/S</PropertyValue>
</PropertyDetail>
</ProductDetail>
</XML>'
SELECT pvt.ProductId, pvt.NodeNum, pvt.[0] AS PKey, pvt.[1] AS PValue
FROM (
SELECT x.XmlCol.value('(ProductId/text())[1]', 'INT') AS ProductId,
(DENSE_RANK() OVER(ORDER BY y.XmlCol) + 1)/2 AS NodeNum,
(DENSE_RANK() OVER(ORDER BY y.XmlCol) + 1) % 2 AS NodeType, -- 0 = PropertyKey, 1 = PropertyValue
y.XmlCol.value('(text())[1]', 'NVARCHAR(50)') AS NodeValue
FROM @x.nodes('XML/ProductDetail') x(XmlCol)
OUTER APPLY x.XmlCol.nodes('PropertyDetail/*') y(XmlCol)
) AS src -- source
PIVOT( MAX(src.NodeValue) FOR NodeType IN ([0], [1]) ) AS pvt
参考:
注意:我不知道这是否可能,但我会因此更改XML模式:
...
<property name="Size" value="XXL" />
<property name="ProdTaxType" value="5%" />
...
。。。
...
实际的XML模式在某种程度上是危险的,因为它依赖于
PropertyKey
/PropertyValue
XML元素的顺序。注意:请不要使用OPENXML。在某些情况下,它会导致内存泄漏。相反,我使用nodes()方法。