【MyBatis 学习二】增删改查 参数占位符 #{} 和 ${}的使用

news2025/1/20 14:50:53

目录

一、增删改查

🌷1、用户类

🌷2、UserMapper

🌷3、UserMapper.xml

🌷4、测试类Test

🌷5、UserService类

🌷6、UserController类

🌷7、注意点总结

二、#{} 和${} 的使用区别

🌷1、#{}与${}的区别(SQL注入)

🌷2、$的SQL注入:用户登录演示

🌷3、$的使用场景:排序+like

🌲 使用场景1:排序

🌲使用场景2:like


一、增删改查

🌷1、用户类

注意:这里的用户类的属性与数据库中的表的字段要一一对应才行。关于两者可以不一致的写法,我们后续再继续介绍~

@Data
public class User {
    private Integer Id;
    private String username;
    private String password;
    private String photo;
    private Date createTime;
    private Date updateTime;
}

🌷2、UserMapper

@Mapper
public interface UserMapper {
    //定义接口:查询数据库内容
    /**
     * 查询所有内容
     * @return 返回数据库中的所有用户
     */
    public abstract List<User> queryAll();

    /**
     * 根据Id查询用户
     * @return 返回符合条件大用户
     */
    //(1)当只有一个参数的时候,可以加注解也可以不加:加注解的话:@Param("uid"),xml文件也叫uid要对应。
    public abstract User queryById(@Param("uid") Integer id);
    //(2)当只有一个参数的时候,如果不加注解@Param,此时的Integer后的参数名叫什么都行,可以与xml文件中的where id = 后的参数名不对应
//    public abstract User queryById(Integer aaa);
    //(3)当有多个参数的时候,就得加注解@Param

    /**
     * 增加:插入一个对象:也就是一条记录
     * @param user 插入对象
     * @return 返回受影响的行数
     */
    public abstract Integer insert(User user);
    //加注解
    public abstract Integer insert1(@Param("aaa") User user);
    //获取自增id
    public abstract Integer insert2(@Param("userinfo")User user);

    /**
     * 修改用户内容:加注解的方式
     * @param user
     */
//    void update(@Param("ccc") User user);
    //不加注解
    void update(User user);

    /**
     * 根据主键删除id:没有加注解
     * @param id
     */
//    void delete(Integer Id);
    //加注解的方式
    void delete(@Param("aaa") Integer id);
}

🌷3、UserMapper.xml

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="demo2.mapper.UserMapper">
<!--表示实现的是demo1下的UserMapper接口-->
    <select id="queryAll" resultType="demo1.model.User">
    select * from userinfo
    </select>
<!--id是方法名,resultType是返回的结果:返回的是一个User对象-->
    
    <select id="queryById" resultType="demo2.model.User">
        select * from userinfo where id = #{uid}
    </select>
    <insert id="insert" useGeneratedKeys="true" keyProperty="id">
        insert into userinfo(username,password,photo) values(#{username},#{password},#{photo})
    </insert>

    <insert id="insert1">
        insert into userinfo(username,password,photo) values(#{aaa.username},#{aaa.password},#{aaa.photo})
    </insert>

<!--    useGeneratedKeys是否使用自动生成生成的key值,赋值的结果返回给谁?-->
    <insert id="insert2" useGeneratedKeys="true" keyProperty="id">
        insert into userinfo(username,password,photo) values(#{userinfo.username},#{userinfo.password},#{userinfo.photo})
    </insert>

<!--    没有使用注解的方式,Mybatis会自动帮我们生成属性,#后面直接加属性名就好了,加注解就是下面这个-->
    <update id="update">
        update userinfo set username=#{ccc.username}, password=#{ccc.password} where id = #{ccc.id}
    </update>

<!--    <delete id="delete">-->
<!--        delete from userinfo where id = #{id}-->
<!--    </delete>-->

    <delete id ="delete">
        delete from userinfo where id= #{aaa}
    </delete>
</mapper>

🌷4、测试类Test

@Slf4j
@SpringBootTest
class UserMapperTest {
    @Autowired
    private UserMapper userMapper;

    @Test
    void queryById() {
        log.info("queryById...");
        User user = userMapper.queryById(1);
        log.info(user.toString());
    }

    @Test
    void insert() {
        log.info("insert...");
        User user = new User();
        user.setUsername("小花花");
        user.setPassword("123");
        user.setPhoto("jfdsf");
        Integer row = userMapper.insert(user);
        log.info("insert:"+row +user.getId());
    }

    @Test
    void insert1() {
        log.info("insert...");
        User user = new User();
        user.setUsername("小叶子");
        user.setPassword("456");
        user.setPhoto("doasj");
        Integer row = userMapper.insert(user);
        log.info("insert:"+row);
    }

    @Test
    void insert2() {
        log.info("insert...");
        User user = new User();
        user.setUsername("小兔子");
        user.setPassword("000");
        user.setPhoto("000");
        System.out.println(user.getId());
        System.out.println(user.getUsername());
        Integer row = userMapper.insert(user);
        log.info("insert:"+row+"自增id"+user.getId());
    }

    @Test
    void update() {
        log.info("update...");
        User user = new User();
        user.setId(1);
        user.setUsername("小老虎");
        user.setPassword("888");
        userMapper.update(user);
    }

    @Test
    void delete() {
        log.info("delete...");
        userMapper.delete(16);
    }
}

🌷5、UserService类

@Service
public class UserService {
    /**
     * 根据Id查询用户
     */
    @Resource
    private UserMapper userMapper;
    public User queryById(){
        return userMapper.queryById(1);
    }
}

🌷6、UserController类

@RestController
@RequestMapping("/web")
public class UserController {
    @Resource
    private UserService userService;
    @RequestMapping("/queryById")
    public User queryById(){
        return userService.queryById();
    }
}

🌷7、注意点总结

1、根据Id查询用户

情况1:当只有一个参数的时候,可以加注解也可以不加。

(1)不加注解:参数名可以任意:

 (2)加注解:@Param("uid"),xml文件也叫uid要对应。而Integer后的名字可以任意。

(3)如果有多个参数:就必须加注解。

2、插入数据

情况1:不加注解

情况2:加注解


问题:怎么拿到自增Id?

(1)useGeneratedKeys:这会令 MyBatis 使⽤ JDBC 的 getGeneratedKeys ⽅法来取出由数据库内部⽣成的主键(⽐如:像 MySQL 和 SQL Server 这样的关系型数据库管理系统的⾃动递增字段),默认值:false。

(2)keyColumn:设置⽣成键值在表中的列名,在某些数据库(像 PostgreSQL)中,当主键列不是表中的第⼀列的时候,是必须设置的。如果⽣成列不⽌⼀个,可以⽤逗号分隔多个属性名称。

(3)keyProperty:指定能够唯⼀识别对象的属性,MyBatis 会使⽤ getGeneratedKeys 的返回值或 insert 语句的 selectKey ⼦元素设置它的值,默认值:未设置(unset)。如果⽣成列
不⽌⼀个,可以⽤逗号分隔多个属性名称。

个人总结:

        其实增删改查都可以加注解或者不加,不过在插入数据和修改数据的时候,加注解@Param("aaa")的方式要用对象名.属性(aaa.getName)获取。在通过id查询和删数数据的时候,给id加注解@Param("uid")的方式,其中注解后的参数要和xml文件中的#{uid}内容一致。 一般不加,防止出错。


二、#{} 和${} 的使用区别

现象1:将queryById的方法中的#替换为$:两者成功!

现象2:根据姓名来查询

🌷1、#{}与${}的区别(SQL注入)

#{}:预编译处理。通过占位方式实现。
${}:字符直接替换。有SQL注入问题

(1)预编译处理是指:MyBatis 在处理#{}时,会将 SQL 中的 #{} 替换为?号,通过占位的方式实现来赋值。

(2)直接替换:是MyBatis 在处理 ${} 时,就是把 ${} 替换成变量的值。直接被当做是sql语句的一部分来执行,被当做是列名了,属于字符的直接替换,会造成SQL注入(用户输入的SQL中带有恶意SQL)。加了引号之后才会被当做一个参数。

举个栗子🌰

我们再打印日志观察一下:

首先在xml文件中添加:

  和上面我们所说的内容是相符的。

问题1:是不是#{}就等同于${}呢?我们继续看下面

🌷2、$的SQL注入:用户登录演示

根据用户名和密码查询:

情况1:根据#查询

 情况2:根据$查询: 

总结1:

sql 注⼊代码:“' or 1='1”。

(1)#方式:

        用户名密码正确,查询出对应用户;

        用户名正确,密码错误,显示null;

        SQL注入:显示为null。

(2)$方式

        用户名密码正确,查询出对应用户;

        用户名正确,密码错误,显示为null;

        SQL注入:也能查询出用户!!!        

回答上面的问题1,两者之间还是不等价的,$存在SQL注入的问题。那么又有新的问题了。

问题2:我们是不是就不使用$了呢?肯定不是,存在肯定有它存在的道理。

🌷3、$的使用场景:排序+like

🌲 使用场景1:排序

如果使用#的方式:发现不能实现排序功能。

 所以在排序问题上只能使用$符号,但是它存在SQL注入(用户输入的SQL带有恶意SQL)的问题,怎么解决呢?那就让用户只能点击,后端在查询之前,对参数进行校验:只能传两个值:desc和asc。其中${}不用再加引号了。

🌲使用场景2:like

like使用#会报错,使用$可以查询出来。

 使用$方式设置,成功。

 但是使用$还是存在SQL注入的问题,我们解决不了,所以用另一种方式:需要使用mysql内置函数concat(字符串直接替换)。

总结2:

        在排序和like的时候,直接使用#会报错,使用$可以成功。但是都存在SQL注入的问题,其中,排序的解决办法是:后端在查询之前,对参数进行校验:只能传两个值:desc和asc。在like的时候,$的SQL注入问题无法解决,我们一般使用mysql内置函数concat。


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/808739.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

一个 SpringBoot 项目能处理多少请求

首先&#xff0c;这个问题有坑&#xff0c;因为 spring boot 不处理请求&#xff0c;只是把现有的开源组件打包后进行了版本适配、预定义了一些开源组件的配置通过代码的方式进行自动装配进行简化开发。这是 spring boot 的价值。 如果我是面试官&#xff0c;我不会问这种问题。…

BLE基础理论/Android BLE开发示例

参考&#xff1a;https://blog.csdn.net/qq_36075612/article/details/127739150?spm1001.2014.3001.5502 参考&#xff1a; https://blog.csdn.net/qq_36075612/article/details/122772966?spm1001.2014.3001.5502 目录 蓝牙的分类传统蓝牙低功耗蓝牙 蓝牙专业词汇&#xff…

深度剖析C++ 异常机制

传统排错 我们早在 C 程序里面传统的错误处理手段有&#xff1a; 终止程序&#xff0c;如 assert&#xff1b;缺陷是用户难以接受&#xff0c;说白了就是一种及其粗暴的手法&#xff0c;比如发生内存错误&#xff0c;除0错误时就会终止程序。 返回错误码。缺陷是需要我们自己…

docker启动容器报错

报错信息 [rootDream soft]# docker run -it -d -p 8080:8080 tomcat eec9fab6b9ca06d2bbf1467aef05d8020ee60448978e10ac20c38888934f0a0b docker: Error response from daemon: driver failed programming external connectivity on endpoint hungry_euclid (163242f0079e72…

C语言之pthread_cond_t信号变化探究总结(八十)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 人生格言&#xff1a; 人生…

MySQL | 常用命令示例

MySQL | 常用命令示例 一、启停MySQL数据库服务二、连接MySQL数据库三、创建和管理数据库四、创建和管理数据表五、数据备份和恢复六、查询与优化 MySQL是一款常用的关系型数据库管理系统&#xff0c;广泛应用于各个领域。在使用MySQL时&#xff0c;我们经常需要编写一些常用脚…

M 芯片的 macos 系统安装虚拟机 centos7 网络配置

centos 安装之前把网络配置配好或者是把网线插好 第一步找到这个 第二步打开网络适配器 选择图中所指位置 设置好之后 开机启动 centos 第三步 开机以后 编写网卡文件保存 重启网卡就可以了&#xff0c;如果重启网卡不管用&#xff0c;则重启虚拟机即可 “ ifcfg-ens160 ” 这…

盖子的c++小课堂——第二十一讲:map

前言 时隔一周&#xff0c;我又来更新了^_^&#xff0c;今天都第二十一讲了&#xff0c;前三个板块马上就结束了&#xff0c;也就是小课堂&#xff08;1&#xff09;马上结束了&#xff0c;敬请期待“盖子的c小课堂&#xff08;2&#xff09;”&#xff0c;嘿嘿~~ map 数据容…

QT--day5(网络聊天室、学生信息管理系统)

服务器&#xff1a; #include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget) {ui->setupUi(this);//给服务器指针实例化空间servernew QTcpServer(this); }Widget::~Widget() {delete ui; …

【C#】.Net Framework框架下的Authorize权限类

2023年&#xff0c;第31周&#xff0c;第3篇文章。给自己一个目标&#xff0c;然后坚持总会有收货&#xff0c;不信你试试&#xff01; 在C#的.NET Framework中&#xff0c;你可以使用Authorize类来处理权限认证。Authorize类位于System.Web.Mvc命名空间中&#xff0c;它提供了…

VS创建wsdl服务提供给java调用

文章目录 前言1.c#创建asp.net web服务1.1 创建ASP.NET Web应用程序1.2 添加服务类1.3 定义服务方法1.3 浏览服务1.4 发布服务1.5 IIS部署服务 2.Java中调用服务2.1 用动态客户端工厂类调用2.1.1 引入依赖2.1.2 调用测试代码2.1.3 测试结果 2.2 创建代理类进行调用2.2.1 使用ws…

微软:向量搜索和向量数据库

向量是未来的数据表示 向量搜索 方法 减少距离计算次数 哈希法空间划分树近邻图 SPTAG 混合了kd树和近邻图 Change 大规律向量搜索 内存可扩展 倒排索引 全局量化进行压缩 top1的召回率比较低 基于图的近邻图 SPANN 倒排索引中的问题&#xff1a; 不平衡的聚类方法低…

Python读取csv、Excel文件生成图表

简介 本文章介绍了通过读取 csv 或 Excel 文件内容&#xff0c;将其转换为折线图或柱状图的方法&#xff0c;并写入 html 文件中。 目录 1. 读取CSV文件 1.1. 生成折线图 1.1.1. 简单生成图表 1.1.2. 设置折线图格式 1.2. 生成柱状图 1.2.1. 简单生成图表 1.2.2. 设置柱…

Python-Python基础综合案例:数据可视化 - 折线图可视化

版本说明 当前版本号[20230729]。 版本修改说明20230729初版 目录 文章目录 版本说明目录知识总览图Python基础综合案例&#xff1a;数据可视化 - 折线图可视化json数据格式什么是jsonjson有什么用json格式数据转化Python数据和Json数据的相互转化 pyecharts模块介绍概况如何…

年薪百万的提示词工程师到底在做什么?

&#x1f3c6; 文章目标&#xff1a;了解热门开源项目 &#x1f340; 入门篇&#xff1a;程序员,必须要知道的热门开源项目! ✅ 创作者&#xff1a;熊猫Jay ✨ 个人公众号: 熊猫Jay字节之旅 (文末有链接) &#x1f341; 展望&#xff1a;若本篇讲解内容帮助到您&#xff0c;请帮…

高忆管理:股票投资策略是什么?有哪些?

在进行股票买卖过程中&#xff0c;出资者需求有自己的方案和出资战略&#xff0c;并且主张严格遵从出资战略买卖&#xff0c;不要跟风操作。那么股票出资战略是什么&#xff1f;有哪些&#xff1f;下面就由高忆管理为我们剖析&#xff1a; 股票出资战略简略来说便是能够协助出资…

左值引用与右值引用的区别?右值引用的意义?

左值引用与右值引用的区别&#xff1f;右值引用的意义&#xff1f; 1 区别1.1 功能差异1.2 左值引用1.3 右值引用1.3.1 实现移动语义1.3.2 实现完美转发 2 引用的作用3 区分左值和右值3.1 左值3.2 右值 1 区别 左值引用是对左值的引用&#xff1b;右值引用是对右值的引用。 &…

【Linux】进程通信 — 共享内存

文章目录 &#x1f4d6; 前言1. 共享内存2. 创建共享内存2.1 ftok()创建key值&#xff1a;2.2 shmget()创建共享内存&#xff1a;2.3 ipcs指令&#xff1a;2.4 shmctl()接口&#xff1a;2.5 shmat()/shmdt()接口:2.6 共享内存没有访问控制&#xff1a;2.7 通过管道对共享内存进…

实验六 调度器-实验部分

目录 一、知识点 1.进程调度器设计的目标 1.1.进程的生命周期 1.2.用户进程创建与内核进程创建 1.3.进程调度器的设计目标 2.ucore 调度器框架 2.1.调度初始化 2.2.调度过程 2.2.1.调度整体流程 2.2.2.设计考虑要点 2.2.3.数据结构 2.2.4.调度框架应与调度算法无关…

二十三章:抗对抗性操纵的弱监督和半监督语义分割的属性解释

0.摘要 弱监督语义分割从分类器中生成像素级定位&#xff0c;但往往会限制其关注目标对象的一个小的区域。AdvCAM是一种图像的属性图&#xff0c;通过增加分类分数来进行操作。这种操作以反对抗的方式实现&#xff0c;沿着像素梯度的相反方向扰动图像。它迫使最初被认为不具有区…