• SQL Server解惑——查询条件IN中能否使用变量


    在SQL Server的查询条件中,能否在IN里面使用变量呢? 如果可以的话,有没有需要注意的地方或一些限制呢?在回答这个问题前,我们先来看看这个例子:

     

    IF EXISTS (SELECT 1 FROM sys.objects WHERE name='TEST' AND type='U')
    BEGIN
        DROP TABLE TEST;
    END
    GO
    CREATE TABLE TEST ( ID INT, NAME VARCHAR(16) );
    GO
     
    INSERT INTO dbo.TEST
    SELECT 1, 'a'  UNION ALL
    SELECT 2, 'b'  UNION ALL
    SELECT 3, 'c'  UNION ALL
    SELECT 4, 'a,b'UNION ALL
    SELECT 5, '''b'',''c''' UNION ALL
    SELECT 6, '''b';
    GO

     

    clip_image001

     

    如下所示,如果查询条件里面,变量只有一个值,此时SQL是正常的。 

    DECLARE @name VARCHAR(16);
    SET @name='a';
     
    SELECT * FROM TEST WHERE name IN (@name);
    GO
     
    DECLARE @name VARCHAR(16);
    SET @name='a,b';
     
    SELECT * FROM TEST WHERE name IN (@name);
    GO

     

    如果我们想在查询条件IN里面输入多个值呢?假如有这样的一个需求,一个变量里面包含b和c的值,现在用'b|c作为条件传入,对其进行拆分为变量'b'和'c', 想查出name=b 和name=c的记录,如下截图所示,SQL其实并没有按你所设想/预想的查出对应记录,而是将ID=5的记录查出来了

     

    DECLARE @name1 VARCHAR(16);
    DECLARE @name2 VARCHAR(16);
    SET @name1='b|c';
    SET @name2=REPLACE(@name1,'|',''',''')
    SELECT @name2
     
    SELECT * FROM TEST WHERE name IN (('''' + @name2 + ''''));

     

     

    clip_image002

     

    下面这个SQL也是同样的结果。 

     

    DECLARE @name1 VARCHAR(16);
    DECLARE @name2 VARCHAR(16);
    SET @name1='b|c';
    SET @name2='''' + REPLACE(@name1,'|',''',''') +''''
    SELECT @name2
     
    SELECT * FROM TEST WHERE name IN (@name2 );

     

    为什么出现了这样的结果呢? 查了大量的官方文档,没有看到关于这个问题的介绍和解释。如果一定要解释上面现象的情况的话,那么是因为SELECT * FROM TEST WHERE name IN (@name2 ); 其实转化为了SELECT * FROM TEST WHERE name =@name2; 也就是说,上面SQL并不会按你所设想的逻辑运算。而是做了一个转换,为什么说是这样的一个转换呢? 当然这也是一个猜想,上面构造的例子也是为了侧面验证这个猜想,另外,上面两个SQL实际执行计划的参数列表(Parameter List)也侧面印证了这个猜想。如果执行计划解析成我们想要的结果,那么Parameter List应该是'b' 和c'

     

     

     

    clip_image003

     

    clip_image004

     

     

     

    解决方案:

     

    1:使用动态SQL

     

    使用动态SQL解决问题,似乎没啥好说的,如下例子所示:

     

    DECLARE @sql_cmd NVARCHAR(max);
    DECLARE @name VARCHAR(16);
     
    SET @name='b|c';
    SET @sql_cmd='SELECT * FROM TEST WHERE name IN (''' + REPLACE(@name,'|',''',''') +''');'
     
    EXEC sp_executesql @sql_cmd;

     

    clip_image005

      

    2:使用临时表或表变量

     

        以这个例子来说,就是将字符串拆分,放入临时表或表变量,然后关联表也好,在IN里面使用子查询也OK。

     

    3:借助STRING_SPLIT() 

    DECLARE @name VARCHAR(16);
     
    SET @name='b|c';
    SELECT *FROM  test WHERE name IN (SELECT value FROM STRING_SPLIT(@name, '|'))

     

    clip_image006

     

    注意STRING_SPLIT函数只有较高版本才支持,SQL Server 2017或SQL Server 2016部分版本支持。

     

    4:借助XML函数来解决问题 

    DECLARE @name VARCHAR(16);
    DECLARE @xml_para XML;
     
    SET @name = 'b|c';
    SET @xml_para = CAST(( '<A>' + REPLACE(@name, '|', '</A><A>') + '</A>' ) AS XML);
     
     
    SELECT  *
    FROM    dbo.TEST
    WHERE   NAME IN ( SELECT    A.value('.', 'varchar(max)') AS [Column]
                      FROM      @xml_para.nodes('A') AS FN ( A ) );

     

    clip_image007

     

    扫描上面二维码关注我
    如果你真心觉得文章写得不错,而且对你有所帮助,那就不妨帮忙“推荐"一下,您的“推荐”和”打赏“将是我最大的写作动力!
    本文版权归作者所有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接.
  • 相关阅读:
    hdu1124 Factorial (求解一个数的阶乘中出现多少个0)
    SQL on Linux: Erro Unable to read instance id from /var/opt/mssql/.system/instance_id
    Error during WebSocket handshake: Unexpected response code: 200 问题处理
    CUP计算资源争抢通过IIS启用处理器关联解决
    ABP在MultipleDbContext也就是多库的场景下发布后异常“Could not find content root folder”问题处理
    ABP运行Login failed for user 'IIS APPPOOL XXXXX Reason: Could not find a login matching the name provided问题解决
    vs2017cpu占用过高解决方案
    docker查看挂载目录Volume
    windows 10安装docker一直挂起在Installing Components and Removing Files
    ABP vue+asp.net core yarn serve报 Cannot find module 'typescript/package.json错误
  • 原文地址:https://www.cnblogs.com/kerrycode/p/14291064.html
Copyright © 2020-2023  润新知