4.动态SQL(if,choose,where,set,trim,foreach遍历)的使用+$和#的区别

news/2024/7/21 10:46:03 标签: sql, 网络, java, intellij-idea, mybatis

文章目录

  • 动态sql
    • 一、动态sql
      • 1.if条件判断
      • 2、choose、when、otherwise
      • 3、where标签
      • 4、set标签
      • 5、`trim标签`
        • 1)替代where标签效果
        • 2) 生成set标签效果
      • 6、foreach迭代遍历
        • 1)属性
      • 7.`SQL`标签-提取重用的SQL代码片段
      • 8、bind标签
      • 9.MyBatis中`${}和#{}`的区别:

sql_1">动态sql

sql_3">一、动态sql

常见动态标签:

  • if
  • choose (when, otherwise)
  • trim (where, set)
  • foreach

1.if条件判断

if标签的test属性判断成立,就会将标签对之间的sql语句拼接到主sql语句上

//如果根据传入参数(emp_name和phone)筛选数据   
<select id="queryByNameAndTel" resultType="employee">
         select emp_name, phone, address, salary
         from employee
         where 1=1
         <if test="empName != null">
             AND emp_name = #{empName}
         </if>
         <if test="phone != null">
               and phone = #{phone}
         </if> 
     </select>

2、choose、when、otherwise

相当与java条件语句Switch语句

//根据多条件筛选语句  
<select id="queryByNameAndTel" resultType="employee">
         select emp_name, phone, address, salary
         from employee
         where 1=1
         <choose>
             <when test="empName != null">
                 and emp_name = #{empName}
             </when>
             <when test="phone != null">
                 and phone = #{phone}
             </when>
             <otherwise>
                 and dept_id = 1
             </otherwise>
         </choose>
     </select>

3、where标签

若子句的开头为 “AND” 或 “OR”,where 元素也会将它们去除。

     <select id="queryByNameAndTel" resultType="employee">
         select emp_name, phone, address, salary
         from employee
         <where>
             <if test="empName != null">
                 and emp_name = #{empName}
             </if>
             <if test="phone != null">
                 and phone = #{phone}
             </if>
         </where>
     </select>

4、set标签

用于动态更新语句的类似解决方案叫做 set,

动态更新需要更新的字段,忽略不更新的字段

 <update id="updateByPrimaryKeySelective">
         update employee
         <set>
             <if test="empName != null">
                 emp_name = #{empName},
             </if>
             <if test="phone != null">
                 phone = #{phone},
             </if>
             <if test="address != null">
                 address = #{address},
             </if>
             <if test="salary != null">
                 salary = #{salary}
             </if>
         </set>
         where id = #{id}
     </update>

5、trim标签

可以使用属性给sql语句添加前缀和后缀删除前缀和后缀

包含属性:

属性说明
prefix(添加前缀)添加前面的关键字(在标签开始位置,添加属性中的内容)
suffix(添加后缀)添加后面的关键字(在标签结束位置,添加属性中的内容)
prefixoverrides(删除前缀)去掉第一个关键字(所有子标签中第一子标签中的前缀关键字)
suffixoverrides(删除后缀)去掉最后一个关键字(所有子标签中最后一个子标签后缀的关键字)
1)替代where标签效果

prefix添加前缀where,代替where标签;prefixOverrides删除子语句判断条件的and | or

  <select id="queryByNameAndTel" resultType="employee">
         select emp_name, phone, address, salary
         from employee
         <trim prefix="where" prefixOverrides="AND|OR">
             <if test="empName != null">
                 and emp_name = #{empName}
             </if>
             <if test="phone != null">
                 and phone = #{phone}
             </if>
         </trim>
     </select>
2) 生成set标签效果

prefix添加前缀set替代set标签,suffixOverrides删除子语句尾部的,

<update id="updateByPrimaryKeySelective">
        update employee
        <trim prefix="set" suffixOverrides=",">
            <if test="empName != null">
                emp_name = #{empName},
            </if>
            <if test="phone != null">
                phone = #{phone},
            </if>
            <if test="address != null">
                address = #{address},
            </if>
            <if test="salary != null">
                salary = #{salary}
            </if>
        </trim>
        where id = #{id}
    </update>

6、foreach迭代遍历

应用于—批量插入或批量删除

1)属性
属性说明
collection集合的名字 默认为collection或者list ,可以通过@Param(“listName”)
item循环出的每个对象 在访问对象属性时,需要加前缀employee.id
open前缀
close后缀
separator以值进行分隔
index索引(当前迭代的序号)
<insert id="addBath">
        insert into dept values
        <foreach collection="deptList" item="dept" open="(" separator="),(" close=")">
          <!--属性值前加入前缀item值-->
            #{dept.deptId},#{dept.deptName},#{dept.remark}
        </foreach>
         <!--           或者这样写也可以
        <foreach collection="deptList" item="dept" separator=",">
         ( #{dept.deptId},#{dept.deptName},#{dept.remark})
       </foreach>
     -->
    </insert>

java%2Ffengyanwen%2F202406221635850.png%20%22image-20240622163538626%22&pos_id=img-MhOCAJD6-1720455002302" alt="外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传" />

7.SQL标签-提取重用的SQL代码片段

通过sql标签封装常使用的字段,通过include标签refid引用其id

 <sql id="base_column_sql">
        account_id ,account_no,account_name,balance
  </sql>
 <select id="queryByNo" resultMap="base-result-account">
        select
        <include refid="base_column_sql"/>
        from account
        where account_no = #{accountNo}
    </select>

8、bind标签

select id, dept_name as deptName, remark from dept where dept_name like #{dn}

9.MyBatis中${}和#{}的区别:

都是用于SQL语句中的占位符?

  1. #参数值直接替换到sql语句,并对参数进行类型转换和字符转义处理;可以有效防止SQL语句注入问题;
  2. $不会进行类型转换和字符转义处理,可能到值sql注入问题
模糊查询为例:
<select id="getBookByNname" resultType="com.example.demo.entity.BookInfo">
    select * from book_info where book_name like '${bookName}';
</select>

<select id="getBookByName" resultType="com.example.demo.entity.BookInfo">
    select * from book_info where book_name like concat('%',#{bookName},'%');
</select>

book_name like ‘${bookName}’;

select * from book_info where book_name like concat('%',#{bookName},'%'); ~~~

http://www.niftyadmin.cn/n/5545203.html

相关文章

云原生监控-Kubernetes-Promethues-Grafana

云原生监控-Prometheus 作者:行癫(盗版必究) 引读:本文章所涉及到技术点包括Prometheus、Grafana、Kuebrnetes;Prometheus基于外部构建采集并监控Kubernetes集群以及集群中的应用,例如使用mysql-node-exporter、nginx-node-exporter采集Kuebrnetes集群中的应用数据,使用…

华为的服务器创新之路

华为作为全球领先的信息与通信技术解决方案供应商&#xff0c;其在服务器领域的创新方法不仅推动了企业自身的发展&#xff0c;也为整个行业的进步做出了重要贡献。以下是华为在服务器领域所采取的一些关键创新方法&#xff1a; 芯片级的自主创新 华为通过自主研发的“鲲鹏”处…

Java版Flink使用指南——将消息写入到RabbitMQ的队列中

大纲 新建工程新增依赖 编码自动产生数据写入RabbitMQ 测试工程代码 在 《Java版Flink使用指南——从RabbitMQ中队列中接入消息流》一文中&#xff0c;我们介绍了如何使用Java在Flink中读取RabbitMQ中的数据&#xff0c;并将其写入日志中。本文将通过代码产生一些数据&#xf…

Visual Studio Code 教程 VsCode安装Live Server以服务形式打开html

搜索Live Server 插件,然后安装 选一个html文件&#xff0c;右键点击 Open with live server,然后就自动弹出来了

软件架构之计算机网络

软件架构之计算机网络 第 4 章 计算机网络4.1 网络架构与协议4.1.1 网络互联模型4.1.2 常见的网络协议4.1.3 IPv6 4.2 局域网与广域网4.2.2 无线局域网4.2.3 广域网技术4.2.4 网络接入技术 4.3 网络互连与常用设备4.4 网络工程4.4.1 网络规划4.4.2 网络设计4.4.3 网络实施 4.5 …

uniapp安卓端实现语音合成播报

最初尝试使用讯飞语音合成方式,能获取到语音数据,但是数据是base64格式的,在安卓端无法播放,网上有说通过转成blob格式的url可以播放,但是uniapp不支持转换的api;于是后面又想其他办法,使用安卓插件播报原生安卓语音播报插件 - DCloud 插件市场 方案一(讯飞语音合成) 1.在讯飞…

vmware lun回收引起的IO问题

起初并没人关注的小问题,正常不过的虚机存储迁移操作,引起的延迟却引发一连串的变化。 环境 vsphere 6.7 + 华为集中式存储 开始 下午5:17 业务反馈,存在数据超时,频繁在1秒钟以内,正常在200ms。需运维排查虚机的状态与IO情况等硬件使用情况。下午5:30 随手翻开zabbix 打开…

imx6ull/linux应用编程学习(14) MQTT基础知识

什么是mqtt&#xff1f; 与HTTP 协议一样&#xff0c; MQTT 协议也是应用层协议&#xff0c;工作在 TCP/IP 四层模型中的最上层&#xff08;应用层&#xff09;&#xff0c;构建于 TCP/IP协议上。 MQTT 最大优点在于&#xff0c;可以以极少的代码和有限的带宽&#xff0c;为连接…