开发过程中动态 SQL 中where 1=1的作用是什么

在这条 MyBatis 动态 SQL 中,where 1=1动态拼接 WHERE 条件的经典技巧 ,核心作用是:解决 "第一个动态条件前是否需要加 AND" 的问题,让条件拼接逻辑更简洁、无歧义

一、先看核心问题:没有 1=1 会怎样?

假设去掉 where 1=1,SQL 会变成:

复制代码
where 
<if test="query.shopIdList != null and query.shopIdList.size()>0">
    and a.shopId in (...)
</if>
<if test="query.shopId != null and query.shopId != ''">
    and a.shopId=#{query.shopId}
</if>
...

如果第一个动态条件不满足 (比如 shopIdList 为 null),最终拼接的 SQL 会是:

复制代码
select ... from dws_shop_itemcate_sum a where and a.shopId=xxx

where 后直接跟 and,触发 SQL 语法错误BadSqlGrammarException)。

二、where 1=1 的核心作用

1=1 是一个恒成立的条件(不会过滤任何数据),但它能:

  1. 占住 where 后的第一个位置 ,所有动态条件都可以统一以 and 开头,无需判断 "是不是第一个条件";
  2. 不影响查询结果1=1 永远为真,不会过滤任何数据,最终的查询结果只由后续的动态条件决定;
  3. 简化逻辑 :开发时不用逐个判断 "当前条件是不是第一个",所有动态条件都按 and 条件 写即可,降低出错概率。

三、代码执行示例(更直观)

场景 1:仅传 shopIdList([9001,9002])

最终拼接的 SQL:

复制代码
select ... from dws_shop_itemcate_sum a 
where 1=1 and a.shopId in (9001,9002)
场景 2:仅传 shopId(9003)+ beginTime(2025-01-01)

最终拼接的 SQL:

复制代码
select ... from dws_shop_itemcate_sum a 
where 1=1 and a.shopId=9003 and a.reportDate>='2025-01-01'
场景 3:不传任何动态条件

最终拼接的 SQL:

复制代码
select ... from dws_shop_itemcate_sum a where 1=1

(等价于 select ... from dws_shop_itemcate_sum a,查询全表)

四、替代方案(MyBatis 更优雅的写法)

MyBatis 提供了 <where> 标签,可以替代 1=1,自动处理多余的 and/or

复制代码
<select id="sumNumberCardDataByShop" resultType="StNumberCardDataBO">
    select 
        <include refid="numberCard_queryColumn"/> 
    from dws_shop_itemcate_sum a  
    <where> <!-- 替代 where 1=1 -->
        <if test="query.shopIdList != null and query.shopIdList.size()>0">
            a.shopId in <!-- 去掉 and -->
            <foreach collection="query.shopIdList" item="shopId"
                index="index" open="(" close=")" separator=",">
                #{shopId}
            </foreach>
        </if>
        <if test="query.shopId != null and query.shopId != ''">
            and a.shopId=#{query.shopId} <!-- 保留 and -->
        </if>
        <if test="query.beginTime != null and  query.beginTime !=''">
            <![CDATA[ and a.reportDate>=#{query.beginTime} ]]>
        </if>
        <if test="query.endTime != null and  query.endTime !=''">
            <![CDATA[ and a.reportDate<=#{query.endTime}]]>
        </if>
    </where>
</select>
<where> 标签的优势:
  • 若有动态条件满足:自动添加 where,并去掉第一个条件前的 and/or
  • 若没有动态条件满足:不生成 where 子句(避免 where 1=1 全表查询);
  • 1=1 更优雅,是 MyBatis 推荐的写法。

五、总结

  1. where 1=1 的本质:用恒真条件占位,让所有动态条件统一以 and 开头,避免语法错误
  2. 优缺点:
    • 优点:简单易懂,新手友好;
    • 缺点:若所有动态条件都不满足,会生成 where 1=1 导致全表查询(可能性能低);
相关推荐
Hello.Reader5 小时前
Flink SQL ANALYZE TABLE手动采集表统计信息,让优化器“更懂数据”
大数据·sql·flink
铅笔小新z5 小时前
【C++】 vector 全面解析:从使用到底层实现
开发语言·c++
没什么本事5 小时前
Springboot CGLIB 代理对象问题
java·spring boot·spring
Javatutouhouduan5 小时前
SpringBoot整合reids之JSON序列化文件夹操作
java·spring boot·spring·bootstrap·html·后端开发·java架构师
曲莫终5 小时前
springboot集成h2内存数据库运行测试用例
数据库·spring boot·测试用例
froginwe115 小时前
Bootstrap4 输入框组
开发语言
她说..5 小时前
Spring AOP场景5——异常处理(附带源码)
java·数据库·后端·spring·springboot·spring aop
listhi5205 小时前
matlab大规模L1范数优化问题
开发语言·matlab
Neolnfra5 小时前
跨站请求伪造攻击(CSRF)解析
服务器·网络·系统安全·网络攻击模型·安全威胁分析·csrf·安全架构