问题查询

Problems querying

我正在尝试按照我一贯的方式查询以下 XML,但由于它的声明和空的 "NextQuantity" - "NextDate" 字段,我遇到了麻烦查询 XML.

正确的做法是什么?

谢谢。

<StocksResp xmlns="http://xxxxxx" xmlns:i="http://www.w3.org/2001/XMLSchema-instance">
  <Stocks>
    <Stock>
      <Sku>30101.06-L</Sku>
      <Quantity>247610</Quantity>
      <NextQuantity1>15243</NextQuantity1>
      <NextDate1>2019-02-27</NextDate1>
      <NextQuantity2 i:nil="true" />
      <NextDate2 />
      <NextQuantity3 i:nil="true" />
      <NextDate3 />
      <NextQuantity4 i:nil="true" />
      <NextDate4 />
      <NextQuantity5 i:nil="true" />
      <NextDate5 />
      <NextQuantity6 i:nil="true" />
      <NextDate6 />
    </Stock>
    <Stock>
      <Sku>30101.06-M</Sku>
      <Quantity>241606</Quantity>
      <NextQuantity1 i:nil="true" />
      <NextDate1 />
      <NextQuantity2 i:nil="true" />
      <NextDate2 />
      <NextQuantity3 i:nil="true" />
      <NextDate3 />
      <NextQuantity4 i:nil="true" />
      <NextDate4 />
      <NextQuantity5 i:nil="true" />
      <NextDate5 />
      <NextQuantity6 i:nil="true" />
      <NextDate6 />
    </Stock>
 </Stocks>
  <Count>4837</Count>
  <Currency i:nil="true" />
  <Language>ES</Language>
  <ErrorCode i:nil="true" />
  <ErrorMessage i:nil="true" />
</StocksResp>

============================================= ==

DECLARE @xml XML
SELECT @xml = x.y FROM OPENROWSET (BULK 'file.xml',SINGLE_BLOB) as x(y)

SELECT  
   x.y.value('Sku[1]','varchar(15)'),
   x.y.value('Quantity[1]','int'),
   x.y.value('NextDate1[1]','varchar(20)'),
   x.y.value('NextQuantity1[1]','int'),
   'mr'
FROM   @xml.nodes('Stocks/Stock') x(y)

这有帮助吗?

DECLARE @xml XML=
N'<StocksResp xmlns="http://xxxxxx" xmlns:i="http://www.w3.org/2001/XMLSchema-instance">
  <Stocks>
    <Stock>
      <Sku>30101.06-L</Sku>
      <Quantity>247610</Quantity>
      <NextQuantity1>15243</NextQuantity1>
      <NextDate1>2019-02-27</NextDate1>
      <NextQuantity2 i:nil="true" />
      <NextDate2 />
      <NextQuantity3 i:nil="true" />
      <NextDate3 />
      <NextQuantity4 i:nil="true" />
      <NextDate4 />
      <NextQuantity5 i:nil="true" />
      <NextDate5 />
      <NextQuantity6 i:nil="true" />
      <NextDate6 />
    </Stock>
    <Stock>
      <Sku>30101.06-M</Sku>
      <Quantity>241606</Quantity>
      <NextQuantity1 i:nil="true" />
      <NextDate1 />
      <NextQuantity2 i:nil="true" />
      <NextDate2 />
      <NextQuantity3 i:nil="true" />
      <NextDate3 />
      <NextQuantity4 i:nil="true" />
      <NextDate4 />
      <NextQuantity5 i:nil="true" />
      <NextDate5 />
      <NextQuantity6 i:nil="true" />
      <NextDate6 />
    </Stock>
 </Stocks>
  <Count>4837</Count>
  <Currency i:nil="true" />
  <Language>ES</Language>
  <ErrorCode i:nil="true" />
  <ErrorMessage i:nil="true" />
</StocksResp>';

-您需要声明默认命名空间。您不需要声明命名空间 i...

WITH XMLNAMESPACES(DEFAULT 'http://xxxxxx','http://www.w3.org/2001/XMLSchema-instance' AS i)
SELECT 
       @xml.value(N'(/StocksResp/Count/text())[1]',N'int') AS StockResp_Count
      ,@xml.value(N'(/StocksResp/Currency/text())[1]',N'int') AS StockResp_Currency 
      ,st.value(N'(Sku/text())[1]',N'nvarchar(max)') AS Stock_Sku
      ,st.value(N'(Quantity/text())[1]',N'int') AS Stock_Quantity
      ,st.value(N'(NextQuantity1/text())[1]',N'int') AS Stock_NextQuantity1
      ,st.value(N'(NextDate1/text())[1]',N'date') AS Stock_NextDate1
      ,st.value(N'(NextQuantity2/text())[1]',N'int') AS Stock_NextQuantity2
      ,st.value(N'(NextDate2/text())[1]',N'date') AS Stock_NextDate2
      ,st.value(N'(NextQuantity3/text())[1]',N'int') AS Stock_NextQuantity3
      ,st.value(N'(NextDate3/text())[1]',N'date') AS Stock_NextDate3
      ,st.value(N'(NextQuantity4/text())[1]',N'int') AS Stock_NextQuantity4
      ,st.value(N'(NextDate4/text())[1]',N'date') AS Stock_NextDate4
      ,st.value(N'(NextQuantity5/text())[1]',N'int') AS Stock_NextQuantity5
      ,st.value(N'(NextDate5/text())[1]',N'date') AS Stock_NextDate5
      ,st.value(N'(NextQuantity6/text())[1]',N'int') AS Stock_NextQuantity6
      ,st.value(N'(NextDate6/text())[1]',N'date') AS Stock_NextDate6
FROM   @xml.nodes('/StocksResp/Stocks/Stock') A(st);

一些备注:

name-number 元素(Date1、Date2 等)是一个非常糟糕的习惯。如果你可以改变这个,你应该使用一组嵌套的元素,并根据它们的位置或像

这样的属性对它们进行编号
<Next index="1">
   <Date>2018-01-01</Date>
   <Quantity>1</Quantity>
</Next>
[... more of them ...]

你可以看到,我直接从变量中读取了 just-once 值,而重复元素是由 [=14] 从派生的 table 中获取的=].