为什么SQL语句Where 1=1 and在SQL Server中不影响性能

SQL性能优化误区
本文讨论了SQL性能优化中常见的误区,特别是对某些规则的教条式应用可能导致的问题,并通过具体的SQL语句分析了查询分析器的工作原理。

    最近一个朋友和我探讨关于Where 1=1 and这种形式的语句会不会影响性能。最后结论是不影响。

    虽然结论正确,但对问题的认识却远远没有解决问题的根本。实际上在T-SQL语句的书写过程中经常犯得错误就是得出一个很窄的结论,然后教条式的奉若圣经,对于T-SQL领域来说,在网上经常可以看到所谓的优化守则,随便在网上搜了一些摘录如下:

  • 不要有超过5个以上的表连接(JOIN)
  • 考虑使用临时表或表变量存放中间结果
  • 少用子查询
  • 视图嵌套不要过深,一般视图嵌套不要超过2个为宜。
  • 对出现在where子句中的字段加索引
  • 避免在索引列上使用函数或计算,在where子句中,如果索引是函数的一部分,优化器将不再使用索引而使用全表扫描
  • 在insert和update维表时都加上一个条件来过滤维表中已经存在的记录
  • 如果使用了IN或者OR等时发现查询没有走索引,使用显式申明指定索引
  • EXISTS要远比IN的效率高。

       ……….

 

问题出在哪了?

    虽然上述指导意见看上去没什么问题,也不能说完全不正确,但实际上有两个重大问题:

    脱离上下文:很多道理只能在一个上下文范围内生效,脱离了上下文范围就毫无意义。举个例子,平常有人对你说你有点肾虚,我想你的第一反应肯定是想办法捍卫男人的尊严了,但如果你去医院检查医生这么说,那你可能就会一脸虔诚的求教如何补了:-),那举上述摘录的语句例子:1)少用子查询,如果在SQL Server操作XML的XPATH按节点属性筛选的时候,那转换成子查询一定会更快 2)如果使用了IN或者OR等时发现查询没有走索引,使用显式申明指定索引,这种情况查询分析器不走索引一定会有其原因,

 

    不解释本质原因:佛语有云“凡所有相,皆是虚妄,若见诸相非相,即见如来”。请看下面故事:

说有一次两个府吏一起来看病,一个叫倪寻,一个叫李延,两人的症状也一样,都是头痛,身上发热,也许都是感冒吧。而华佗却说:“倪寻应当用下法来治,李延应当用汗法来治(寻当下之,延当发汗)。”旁人认为很奇怪,大家也一定认为很奇怪吧,为什么同样的一个病,同样的症状,会有不同的治疗法子呢?华佗解释了,他说:“倪寻是外实,而立延是内实,所以用了不同的法子。”果然,第二天,他们两的病都好了。

    其实可以看出,完全同样的症状,可以是完全不同的原因,反之,同样的原因,也可以形成完全不同的“相”。如果仅仅是看到“相”而采取应激处理措施,往往结果会不尽人意。

 

Think Like Query Optimizer

    在每一个领域都有其领域内的规则,最简单来说,如果你不符合C#规范去编程,比如错误的使用关键字,那么编译就会报错。当然,每一个领域内还会有一些隐藏的规则,也有人会说是所谓的“潜规则”,这类规则往往不在明面上,比如说你不符合最佳实践编写一段程序,编译不会报错,但因此而引起的性能或是安全性问题就是你需要遵循最佳实践这个“潜规则”才能避免。

    而在SQL Server领域,T-SQL语句到查询结果返回需要经历一个完整的周期,如图1:

    image

    图1.T-SQL生命周期

 

    因此,在关系数据库领域,SQL语句的写法只是一个抽象的逻辑,而不是像编程语言那样直接的实现。比如说访问一行数据,如果是编程语言实现,就需要指定连接数据的方式,打开数据,按某个方式取出数据,最后还要关闭连接,而在SQL Server中,T-SQL仅仅是定义如何去获取所需的数据,而无需考虑实现细节。

    图1中从T-SQL到具体返回数据经历了多个步骤,每一个步骤又存在大量的规则。因此在本文提到Where 1=1 and引起的性能问题就需要按照查询分析器的规则去考虑为什么,这也是Think like query optimizer。

 

    在SQL Server中,T-SQL需要编译为执行计划才能去执行,在编译过程中,Query Optimizer需要考虑很多元数据,比如说表上的索引、数据分布、估计行数、一些参数配置、硬件环境等,在这其中,最重要的就是估计行数,SQL Server需要估计行数来估计成本。

 

Where 1=1 and写法为什么不会变慢?

    因为查询分析器在代数树优化阶段就把1=1 直接给过滤掉了。这个功能就是查询优化器中所谓的“Constant Folding”。

    我们这里假设查询分析器在代数树优化阶段没有把where 1=1这种情况直接过滤掉。

    比如语句select * from table where a=1 and b=2 这个语句,SQL Server估计的行数会是:

    a列的选择率*b列的选择率*表中采样的总行数

    因此,当Where 1=1 and a=1时,结果就变为

     1*a列的选择率 *表中采样的总行数=a列的选择率 *表中采样的总行数

 

    因此无论是否有1=1 and,查询分析器都会估计相同的行数,从而拥有同样的执行计划,因此不影响性能。

 

    当我们明白了查询分析器对A and B这种写法是如何估计行数之后,那么我们就可以推算出什么情况A and B可能引起执行计划不准确。从公式来看,SQL Server认为A列和B列是无关联的,如果A和B关联很大,那么估计的行数一定会非常不准。

    这里我们举例,假如表中有100万行数据,where a=1的数据有1万条,where b=1的数据有1万条,则A和B的选择性都是1/100=0.01,在Where中A And B联合的估计行数则变为0.01*0.01=0.0001*100万=100行,假设where a=1 和b=1所筛选的数据为同样的1万行数据,则估计行数为100而实际行数为1万,则可能引起执行计划的不准确,从而引起性能问题。当然,这种情况的确是少数,但发生后往往对性能有一定影响,因此SQL Server 2014新的行数估计采用了指数退让算法,在这种情况下就会估计为1000行,从而引起性能问题的可能性会变小,2014指数退让算法不是本文的重点,因此也不多讲了。

转载于:https://www.cnblogs.com/CareySon/p/4138575.html

<think>好的,用户问的是SQL语句where 1=1是什么意思。首先,我得回忆一下自己以前学过的SQL知识,记得在动态生成查询条件的时候,有时候会看到这样的写法。过具体为什么要用1=1呢?可能用户看到别人的代码里有这个写法,太明白它的作用,或者担心是否有安全问题。 首先,我需要确认1=1SQL中的含义。1=1显然是一个恒真的条件,也就是说无论什么情况下,这个条件都会成立。那为什么要在where子句里加一个永远为真的条件呢?这可能和动态拼接SQL语句有关。比如在编程时,可能需要根据同的条件添加同的查询条件,这时候如果在基础语句里加上where 1=1,后面就可以统一地用and来连接其他条件,而需要判断是否是第一个条件,避免出现语法错误。 比如,如果用户有多个可选条件,比如年龄、姓名、城市,这些条件可能都存在,也可能部分存在。如果没有where 1=1,那么第一个条件前需要加where,后面的条件前加and,这样处理起来比较麻烦,容易出错。而有了where 1=1,后面所有的条件都可以直接用and连接,简化代码逻辑。 接下来,我需要考虑用户可能的疑问,比如这样做是否会影响性能?一般来说,数据库的优化器会识别1=1这样的恒真条件,并自动忽略,所以性能会有影响。但可能需要提到这一点,让用户放心。 另外,用户可能担心这样的写法是否存在安全隐患,比如SQL注入。where 1=1本身并会导致注入问题,关键在于如何拼接参数。如果直接拼接用户输入,管有没有1=1都会有风险,所以还是要使用参数化查询来防止注入。 可能还需要举一个具体的例子,比如在Java或Python中动态生成查询的情况,说明如何利用where 1=1来简化条件拼接。这样用户更容易理解应用场景。 总结一下,回答的结构应该是先解释1=1的作用,然后说明它的用途,特别是在动态SQL中的应用,接着可能涉及性能和安全性问题,最后给出示例。确保用户仅知道这个写法的意思,还能明白它的实际应用和注意事项。</think>在SQL语句中,`WHERE 1=1` 是一个看似冗余但实际有特定用途的写法。以下是逐步解释: --- ### 1. **基础含义** - `1=1` 是一个**恒真条件**(总是返回`True`),因此 `WHERE 1=1` 会对查询结果产生任何过滤作用。 - 例如: ```sql SELECT * FROM users WHERE 1=1; ``` 等价于: ```sql SELECT * FROM users; -- 没有过滤条件 ``` --- ### 2. **实际用途** 主要目的是**简化动态SQL的拼接逻辑**,尤其在编程中需要根据条件动态生成查询语句时: #### 场景示例 假设需要根据用户输入动态拼接多个条件: ```sql SELECT * FROM users WHERE 1=1 {AND age > 18} -- 可选条件1 {AND name LIKE '%张%'} -- 可选条件2 {AND city = '北京'} -- 可选条件3 ``` #### 优势 - **统一拼接格式**:所有条件都可以直接以 `AND` 开头,无需判断是否是第一个条件。 - **避免语法错误**:如果所有条件都未选中,语句会因缺少 `WHERE` 或多余 `AND` 报错。 --- ### 3. **性能影响** - **无额外开销**:数据库优化器会自动忽略恒真条件 `1=1`,影响查询性能。 - **执行计划**:最终生成的执行计划与直接写 `WHERE` 条件完全相同。 --- ### 4. **其他变体** - `WHERE 1=0`:用于快速清空表结构(如 `CREATE TABLE new_table AS SELECT * FROM old_table WHERE 1=0`)。 - 类似写法:`WHERE TRUE`(在支持布尔类型的数据库中)。 --- ### 5. **注意事项** - **避免滥用**:静态SQL中无需使用 `WHERE 1=1`,仅动态拼接时才有意义。 - **防SQL注入**:动态拼接时仍需使用参数化查询,能直接拼接用户输入。 --- ### 总结 `WHERE 1=1` 是动态SQL中的一种编程技巧,通过提供“锚点”简化条件拼接逻辑,实际查询时会被数据库自动优化,影响性能
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值