7
    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 Server2008r2で細断処理しようとしているXMLの表現。「タイトル」ノードを見つけて、製品カテゴリにある必要な値を取得します。ここで「ソリューション番号」を取得したいと思いますが、特定の製品「グループ」があるため、これは製品の上の1つ以上の親ノードである可能性があります。

親ノードを見つけるまで、名前( "solutionnumber")で確認するにはどうすればよいですか?助けてくれてありがとう。

4

4 に答える 4

3

私の知識に直接的な方法はありません。ただし、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>は実際には、祖先の1つの兄弟であり、祖先自体ではないことに注意してください。

このソリューションでは、事前に最大深度を知っておく必要があります。


後方ではなく前方に進む場合は、このソリューションを使用することもできます。

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>タグが>タグの直接の子であるという事実を使用し<solutionます。最初にすべての<solution>タグが見つかります。そのすべてのタイトルの子孫よりも、クロスアプライで見つかります。ノードでnodes関数を使用できないため、その間に「query('。')」の計算があります。

上記の解決策以外に、これは>タグと<solution>タグの間の任意の距離を処理でき<titleます。

于 2012-10-27T01:54:08.700 に答える
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)
于 2012-10-28T02:41:48.710 に答える
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);
于 2012-10-27T02:09:32.150 に答える
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)
于 2015-07-14T04:53:37.857 に答える