declare @xmlsample xml =
'<root>
<solution>
<solutionnumber>1</solutionnumber>
<productgroup>
<productcategory>
<price>100</price>
<title>Some product</title>
<tax>1</tax>
</productcategory>
</productgroup>
<productcategory2>
<price>200</price>
<title>Some other product</title>
<tax>2</tax>
</productcategory2>
</solution>
<solution>
<solutionnumber>2</solutionnumber>
<productcategory2>
<price>200</price>
<title>Some other product</title>
<tax>2</tax>
</productcategory2>
</solution>
</root>'
SELECT
--T.C.value('(./ancestor::ns1:solutionNumber)[1]', 'varchar(50)') AS solutionnumber ?? no clue
T.C.value('(price)[1]', 'numeric(18,2)') AS price
,T.C.value('(title)[1]', 'varchar(50)') AS title
,T.C.value('(tax)[1]', 'numeric(18,2)') AS tax
FROM @xmlsample.nodes('//node()[title]') AS T(C)
我试图在SQL Server 2008 r2中粉碎的XML的表示。我找到了“标题”节点,并获取了产品类别中我需要的值。现在我想获得“解决方案编号”,但这可能是产品上方的一个或多个父节点,因为有某些产品“组”。
在找到之前,我将如何检查父节点的名称(“solutionnumber”)?感谢您的帮助。
答案 0 :(得分:2)
我的知识没有直接的方法。但是,您可以使用COALESCE进行搜索:
SELECT
COALESCE(T.C.value('../solutionnumber[1]', 'INT'),
T.C.value('../../solutionnumber[1]', 'INT'),
T.C.value('../../../solutionnumber[1]', 'INT')) solutionnumber,
T.C.value('(price)[1]', 'numeric(18,2)') AS price,
T.C.value('(title)[1]', 'varchar(50)') AS title,
T.C.value('(tax)[1]', 'numeric(18,2)') AS tax
FROM
@xmlsample.nodes('//node()[title]') AS T ( C )
请注意<solutionnumber
&gt;是一个祖先的兄弟,而不是祖先本身。
此解决方案要求您提前知道最大深度。
如果您宁愿前进而不是倒退,也可以使用此解决方案:
SELECT solutionNodes.solutionNode.value('solutionnumber[1]','INT') AS solutionnumber,
T.C.value('(price)[1]', 'numeric(18,2)') AS price,
T.C.value('(title)[1]', 'varchar(50)') AS title,
T.C.value('(tax)[1]', 'numeric(18,2)') AS tax
FROM @xmlsample.nodes('//solution') AS solutionNodes (solutionNode)
CROSS APPLY (SELECT solutionNodes.solutionNode.query('.')) solutions(solutionXML)
CROSS APPLY solutions.solutionXML.nodes('//node()[title]') T ( C )
它使用<solutionnumber
&gt;的事实。标签是<solution
&gt;的直接子项。标签。首先是<solution
&gt;标签被找到。比所有的标题后代都有交叉申请。因为您无法在节点上使用节点函数,所以在它们之间会计算“query('。')”。
除了上述解决方案之外,这个解决方案可以处理<solution
&gt;之间的任何距离。标记为<title
&gt;标签
答案 1 :(得分:2)
也许我正在倒退。多次交叉申请将完成这项工作。感谢另一个论坛的一些帮助。
SELECT
--T.C.value('(./ancestor::ns1:solutionNumber)[1]', 'varchar(50)') AS solutionnumber ?? no clue
m.c.value('(solutionnumber)[1]', 'int') as solutionnumber
,T.C.value('(price)[1]', 'numeric(18,2)') AS price
,T.C.value('(title)[1]', 'varchar(50)') AS title
,T.C.value('(tax)[1]', 'numeric(18,2)') AS tax
FROM @xmlsample.nodes ('//solution') as m (c)
cross apply m.c.nodes ('.//node()[title]') as t(C)
答案 2 :(得分:1)
SQL Server不支持向后追溯到祖先,所以这里是一个迂回的方式caching
指向祖先的指针,同时降序为XML。
declare @xmlsample xml =
'<root>
<solution>
<solutionnumber>1</solutionnumber>
<productgroup>
<productcategory>
<price>100</price>
<title>Some product</title>
<tax>1</tax>
</productcategory>
</productgroup>
<productcategory2>
<price>200</price>
<title>Some other product</title>
<tax>2</tax>
</productcategory2>
</solution>
<solution>
<solutionnumber>2</solutionnumber>
<productcategory2>
<price>200</price>
<title>Some other product</title>
<tax>2</tax>
</productcategory2>
</solution>
</root>';
WITH Xml_CTE AS
(
SELECT node.query('*') AS children,
node.value('fn:local-name(.)','varchar(100)') localName,
node.exist('title') IsTitleParent,
CAST(null as xml) as solution
FROM @xmlsample.nodes('/*') AS root(node)
UNION ALL
SELECT node.query('*') AS children,
node.value('fn:local-name(.)','varchar(100)') localName,
node.exist('title') IsTitleParent,
CASE WHEN node.value('fn:local-name(.)', 'varchar(100)') = 'solution'
THEN node.query('.')
ELSE solution END
FROM Xml_CTE x
CROSS APPLY x.children.nodes('*') AS child(node)
)
SELECT solution.value('(solution/solutionnumber/text())[1]', 'int') solutionNumber
,children.value('(price)[1]', 'numeric(18,2)') price
,children.value('(title)[1]', 'varchar(50)') title
,children.value('(tax)[1]', 'numeric(18,2)') tax
FROM Xml_CTE
WHERE IsTitleParent = 1 -- matches .nodes('//node()[title]')
OPTION (MAXRECURSION 0);
答案 3 :(得分:0)
这将完美地运作......
Declare @SomeXML XML
SET @SomeXML = '<SomeValue>GGGG</SomeValue><SomeValue>MMMM</SomeValue><SomeValue>AAA</SomeValue>'
select ROW_NUMBER() over ( order by b), b.value('.', 'varchar(50)')
from @SomeXML.nodes('(/SomeValue)') AS a(b)