2

我认为以下查询假设不返回任何内容,但相反,它返回一条记录,其中包含一列null

select *
from ( select 1 as "data"
       where 0 = 1
       for xml path('row') ) as fxpr(xmlcol)

如果您只运行子查询 - 不会返回任何内容,但是当此子查询有外部查询select时,null将返回对其执行 a 。

为什么会这样?

4

2 回答 2

0

XML数据在 SQL Server 中被视为NULL 。

select *
from ( select 1 as "data"
       where 0 = 1
       for xml path('row') ) as fxpr(xmlcol)

子查询将首先执行,子查询的结果,即(空行集)将被转换为XML,因此得到NULL值。

于 2018-05-17T15:20:00.260 回答
0

SQL Server 将尝试预测类型。看这个

SELECT tbl.[IsThereAType?] + '_test'
      ,tbl.ThisIsINT       + 100
FROM
(
    SELECT NULL AS [IsThereAType?]
          ,3 AS ThisIsINT
    UNION ALL 
    SELECT 'abc'
          ,NULL
    --UNION ALL 
    --SELECT 1  
    --    ,NULL
) AS tbl;

第一列将被预测为字符串类型,而第二列将被视为INT. 这就是+顶部操作员工作的原因。尝试将数字添加到第一个或字符串添加到第二个。这将失败。

尝试取消注释最后一个块,它也会失败。

预测是在很早的阶段完成的。看看这个,我确实包含了第三个UNION ALL(无效查询,破坏类型):

EXEC sp_describe_first_result_set 
N'SELECT *
FROM
(
    SELECT NULL AS [IsThereAType?]
          ,3 AS ThisIsINT
    UNION ALL 
    SELECT ''abc''
          ,NULL
    UNION ALL 
    SELECT 1
          ,NULL

) AS tbl';

结果返回“IsThereAType?” 作为INT!(我很确定这是相当随机的,并且在您的系统上可能会有所不同。)
顺便说一句:如果没有最后一个块,类型是VARCHAR(3)......

现在回答你的问题

XML 被视为NTEXT(尽管已弃用!)并且需要预测,TYPEXML

EXEC sp_describe_first_result_set N'SELECT ''blah'' FOR XML PATH(''blub'')';
EXEC sp_describe_first_result_set N'SELECT ''blah'' FOR XML PATH(''blub''),TYPE';

包裹在子选择中的相同返回为NVARCHAR(MAX)resp。XML

EXEC sp_describe_first_result_set N'SELECT * FROM(SELECT ''blah'' FOR XML PATH(''blub'')) AS x(y)';
EXEC sp_describe_first_result_set N'SELECT * FROM(SELECT ''blah'' FOR XML PATH(''blub''),TYPE) AS x(y)';

嗯,这实际上有点奇怪...... XML 是一个标量值,取为NTEXT,NVARCHAR(MAX)XML(取决于您调用它的方式)。但不允许在子选择中放置裸标量:

SELECT * FROM('blah') AS x(y) --fails

虽然这没关系

SELECT * FROM(SELECT 'blah') AS x(y)    

结论:

在您的特殊情况下,查询解析器似乎略有不一致:

尽管子选择不能仅包含一个标量值,但SELECT ... FOR XML(实际上返回标量)不会被拒绝。引擎似乎将此解释为SELECT 返回一个标量值。这完全没问题。

这对于将嵌套子选择作为列(相关子查询)嵌套 XML 很有用:

SELECT TOP 5 t.TABLE_NAME
            ,(
              SELECT COLUMN_NAME,DATA_TYPE 
              FROM INFORMATION_SCHEMA.COLUMNS AS c
              WHERE c.TABLE_SCHEMA=t.TABLE_SCHEMA
                AND c.TABLE_NAME=t.TABLE_NAME
              FOR XML PATH('Column'),ROOT('Columns'),TYPE
             ) AS AllTablesColumns
FROM INFORMATION_SCHEMA.TABLES AS t;

如果没有该FOR XML子句,这将失败(...多个值... / ...只有一列...

SELECT将泛型作为参数传递?

有人会说这是不可能的,但你可以试试这个:

CREATE FUNCTION dbo.TestType(@x XML)
RETURNS TABLE
AS
RETURN
SELECT @x AS BringMeBack;
GO

--The SELECT must be wrapped in paranthesis!
SELECT *
FROM dbo.TestType((SELECT TOP 5 * FROM sys.objects FOR XML PATH('x'),ROOT('y')));
GO

DROP FUNCTION dbo.TestType;
于 2018-05-18T08:33:06.470 回答