公用表(CTE)表达式引发的改变执行顺序同WHERE条件顺序引发的bug

以下模拟一下CTE出错

/*测试环境
Microsoft SQL Server 2008 R2 (RTM) - 10.50.1600.1 (X64)   Apr  2 2010 15:48:46   Copyright (c) Microsoft Corporation  Developer Edition (64-bit) on Windows NT 6.1 <X64> (Build 7601: Service Pack 1)
*/

生成表Tab数据:

--> --> (Roy)生成測試數據

if not object_id(‘Tab‘) is null
    drop table Tab
Go
Create table Tab([Col1] int,[COl2] nvarchar(5))
Insert Tab
select 1,N‘a,b,c‘ union all
select 2,N‘d,e‘ union all
select 3,N‘f‘
Go

方法1:用CTE引发函数出错

if object_id(‘Tempdb..#Tab‘) is not null
    drop table #Tab
select top 100 ID=Identity(int,1,1) into #Tab from syscolumns a,syscolumns b

declare @Str varchar(10)=‘a‘

;with Cte
as
(
Select
    a.Col1,COl2=substring(a.Col2,b.ID,charindex(‘,‘,a.Col2+‘,‘,b.ID)-b.ID)
from
    Tab a,#Tab b
where
    charindex(‘,‘,‘,‘+a.Col2,b.ID)=b.ID --也可用 substring(‘,‘+a.COl2,b.ID,1)=‘,‘
)
select Col1 from Cte where @str=COl2
/*
消息 537,级别 16,状态 3,第 8 行
传递给 LEFT 或 SUBSTRING 函数的长度参数无效。
*/

方法2:直接用语句时不会报错:

if object_id(‘Tempdb..#Tab‘) is not null
    drop table #Tab
select top 100 ID=Identity(int,1,1) into #Tab from syscolumns a,syscolumns b

declare @Str varchar(10)=‘a‘

Select
    a.Col1
from
    Tab a,#Tab b
where
    charindex(‘,‘,‘,‘+a.Col2,b.ID)=b.ID --也可用 substring(‘,‘+a.COl2,b.ID,1)=‘,‘
and substring(a.Col2,b.ID,charindex(‘,‘,a.Col2+‘,‘,b.ID)-b.ID) [email protected]

/*
Col1
1
*/

方法3:把Where条件换一下顺序也出错

if object_id(‘Tempdb..#Tab‘) is not null
    drop table #Tab
select top 100 ID=Identity(int,1,1) into #Tab from syscolumns a,syscolumns b
go

declare @Str varchar(10)=‘a‘

Select
    a.Col1
from
    Tab a,#Tab b
where
    substring(a.Col2,b.ID,charindex(‘,‘,a.Col2+‘,‘,b.ID)-b.ID) [email protected] and charindex(‘,‘,‘,‘+a.Col2,b.ID)=b.ID --也可用 substring(‘,‘+a.COl2,b.ID,1)=‘,‘

查原因从执行计划来找

if object_id(‘Tempdb..#Tab‘) is not null
    drop table #Tab
select top 100 ID=Identity(int,1,1) into #Tab from syscolumns a,syscolumns b
go
SET SHOWPLAN_ALL ON;
go
--a.查看方法1执行计划

declare @Str varchar(10)=‘a‘

;with Cte
as
(
Select
    a.Col1,COl2=substring(a.Col2,b.ID,charindex(‘,‘,a.Col2+‘,‘,b.ID)-b.ID)
from
    Tab a,#Tab b
where
    charindex(‘,‘,‘,‘+a.Col2,b.ID)=b.ID --也可用 substring(‘,‘+a.COl2,b.ID,1)=‘,‘
)
select Col1 from Cte where @str=COl2
go
--b.查看方法2执行计划

declare @Str varchar(10)=‘a‘

Select
    a.Col1
from
    Tab a,#Tab b
where
    charindex(‘,‘,‘,‘+a.Col2,b.ID)=b.ID --也可用 substring(‘,‘+a.COl2,b.ID,1)=‘,‘
and substring(a.Col2,b.ID,charindex(‘,‘,a.Col2+‘,‘,b.ID)-b.ID) [email protected]

go

SET SHOWPLAN_ALL off
go

方法1、方法2生成的执行计划图

注意看以上第4行的运算,再查第3行执行顺序
方法1:可看到语句先执行条件(@str=COl2),再执行(charindex(‘,‘,‘,‘+a.Col2,b.ID)=b.ID)

方法2:可看到语句先执行条件(charindex(‘,‘,‘,‘+a.Col2,b.ID)=b.ID),再执行(substring(a.Col2,b.ID,charindex(‘,‘,a.Col2+‘,‘,b.ID)-b.ID) [email protected])

方法3同方法1一样原因,条件的顺序也会引发执行出错

时间: 2024-10-09 23:36:25

公用表(CTE)表达式引发的改变执行顺序同WHERE条件顺序引发的bug的相关文章

SQL中使用WITH AS提高性能,使用公用表表达式(CTE)简化嵌套SQL

一.WITH AS的含义     WITH AS短语,也叫做子查询部分(subquery factoring),可以让你做很多事情,定义一个SQL片断,该SQL片断会被整个SQL语句所用到.有的时候,是为了让SQL语句的可读性更高些,也有可能是在UNION ALL的不同部分,作为提供数据的部分. 特别对于UNION ALL比较有用.因为UNION ALL的每个部分可能相同,但是如果每个部分都去执行一遍的话,则成本太高,所以可以使用WITH AS短语,则只要执行一遍即可.如果WITH AS短语所定

Sql — CTE公用表表达式和With用法总结

CTE(Common Table Expression) 公用表表达式,它是在单个语句的执行范围内定义的临时结果集,只在查询期间有效.它可以自引用,也可在同一查询中多次引用,实现了代码段的重复利用. CTE最大的好处是提升T-Sql代码的可读性,可以更加优雅简洁的方式实现递归等复杂的查询. CTE可用于: ⒈ 创建递归查询,这个应该是CTE最好用的地方 ⒉ 在同一语句中多次引用生成的表 3. 减少子查询和表变量,提高执行效率 CTE优点: 1. 使用 CTE 可以获得提高可读性和轻松维护复杂查询

T-SQL 公用表表达式(CTE)

公用表表达式(CTE) 在编写T-SQL代码时,往往需要临时存储某些结果集.前面我们已经广泛使用和介绍了两种临时存储结果集的方法:临时表和表变量.除此之外,还可以使用公用表表达式的方法.公用表表达式(Common Table Expression)是SQL Server2005版本的引入的一个特性.CTE可以看组是一个临时的结果集,可以再接下来来的一个SELECT,INSERT,UPDATE,DELETE,MERGE语句中多次引用.使用公用表达式CTE可以让语句更加清晰简练.与公用表达式作用类似

关于SQL中CTE(公用表表达式)(Common Table Expression)的总结

WITH AS短语,也叫做子查询部分(subquery factoring),可以让你做很多事情,定义一个SQL片断,该SQL片断会被整个SQL语句所用到 一.WITH AS的含义 WITH AS短语,也叫做子查询部分(subquery factoring),可以让你做很多事情,定义一个SQL片断,该SQL片断会被整个SQL语句所用到.有的时候,是为了让SQL语句的可读性更高些, 也有可能是在UNION ALL的不同部分,作为提供数据的部分. 特别对于UNION ALL比较有用.因为UNION

T-SQL 之 公用表表达式(CTE)

在编写T-SQL代码时,往往需要临时存储某些结果集.在CTE之前常用的两种临时存储结果集为:临时表和表变量.除此之外,还可以使用公用表表达式的方法. 公用表表达式(Common Table Expression)是SQL Server2005版本的引入的一个特性.CTE可以看作是一个临时的结果集,可以再接下来来的一个SELECT,INSERT,UPDATE,DELETE,MERGE 语句中多次引用.使用公用表达式CTE可以让语句更加清晰简练. 一.三种方式的对比 (一).临时表:需要在临时数据库

SQL中使用WITH AS提高性能-使用公用表表达式(CTE)简化嵌套SQL

转:http://wudataoge.blog.163.com/blog/static/80073886200961652022389/ 一.WITH AS的含义     WITH AS短语,也叫做子查询部分(subquery factoring),可以让你做很多事情,定义一个SQL片断,该SQL片断会被整个SQL语句所用到.有的时候,是为了让SQL语句的可读性更高些, 也有可能是在UNION ALL的不同部分,作为提供数据的部分. 特别对于UNION ALL比较有用.因为UNION ALL的每

使用公用表表达式(CTE)简化嵌套SQL

一.WITH AS的含义     WITH AS短语,也叫做子查询部分(subquery factoring),可以让你做很多事情,定义一个SQL片断,该SQL片断会被整个SQL语句所用到.有的时候,是为了让SQL语句的可读性更高些,也有可能是在UNION ALL的不同部分,作为提供数据的部分. 特别对于UNION ALL比较有用.因为UNION ALL的每个部分可能相同,但是如果每个部分都去执行一遍的话,则成本太高,所以可以使用WITH AS短语,则只要执行一遍即可.如果WITH AS短语所定

SQL Server 公用表表达式(CTE)实现递归

公用表表达式简介: 公用表表达式 (CTE) 可以认为是在单个 SELECT.INSERT.UPDATE.DELETE 或 CREATE VIEW 语句的执行范围内定义的临时结果集.CTE 与派生表类似,具体表现在不存储为对象,并且只在查询期间有效.与派生表的不同之处在于,公用表表达式 (CTE) 具有一个重要的优点,那就是能够引用其自身,从而创建递归 CTE.递归 CTE 是一个重复执行初始 CTE 以返回数据子集直到获取完整结果集的公用表表达式. 下面先创建一个表,并插入一些数据: crea

SQL.WITH AS.公用表表达式(CTE)(转)

一.WITH AS的含义    WITH AS短语,也叫做子查询部分(subquery factoring),可以让你做很多事情,定义一个SQL片断,该SQL片断会被整个SQL语句所用到.有的时候,是为了让SQL语句的可读性更高些,也有可能是在UNION ALL的不同部分,作为提供数据的部分.特别对于UNION ALL比较有用.因为UNION ALL的每个部分可能相同,但是如果每个部分都去执行一遍的话,则成本太高,所以可以使用WITH AS短语,则只要执行一遍即可.如果WITH AS短语所定义的