【JavaEE进阶】——Mybatis操作数据库(使用注解和XML方式)

目录

🚩三层架构 

🎈JDBC操作回顾

🚩什么是MyBatis

🚩MyBatis⼊⻔

🎈准备工作

📝创建⼯程

📝数据准备

🎈配置数据库连接字符串

🎈写持久层代码

🎈单元测试

🚩MyBatis的基础操作

🎈打印日志

🎈参数传递

🎈增(Insert)

📝返回主键(实现自增效果)

🎈删(delete)

🎈改(Update)

🎈查(Select)

📝起别名

📝结果映射

📝开启驼峰命名(推荐)

🚩MyBatis  XML配置文件

🎈配置连接字符串和MyBatis

🎈写持久层代码

📝添加 mapper 接⼝

📝添加 UserInfoXMLMapper2.xml

📝查询所有⽤⼾的具体实现 :

📝单元测试

🎈增删改查操作

📝增(Insert)

📝删(Delete)

📝改(Update)

📝查(Select)


本节⽬标
  • 1. 使⽤MyBatis完成简单的增删改查操作, 参数传递.
  • 2. 掌握MyBatis的两种写法: 注解 和 XML⽅式
  • 3. 掌握MyBatis 相关的⽇志配置

🚩三层架构 

在应⽤分层学习时, 我们 了解到web应⽤程序⼀般分为三层,即:Controller、Service、Dao .
  • 表现层:Controller  参数接收,参数校验,参数格式转换,结果格式转换,结果返回等等
  • 业务逻辑层:Service  介于表现层和数据层之间,业务逻辑相关的处理
  • 数据层:Dao  与数据相关的,数据的增删改查
之前的案例中,请求流程如下:
浏览器发起请求, 先请求Controller, Controller接收到请求之后, 调⽤ Service进⾏业务逻辑处理, Service再调⽤Dao, 但是Dao层的数据是Mock的, 真实的数据应该从数据库中读取.
我们学习MySQL数据库时,已经学习了JDBC来操作数据库, 但是JDBC操作太复杂了.

🎈JDBC操作回顾

我们先来回顾⼀下 JDBC 的操作流程:
  • 1. 创建数据库连接池 DataSource
  • 2. 通过 DataSource 获取数据库连接 Connection
  • 3. 编写要执⾏带 ? 占位符的 SQL 语句
  • 4. 通过 Connection 及 SQL 创建操作命令对象 Statement
  • 5. 替换占位符:指定要替换的数据库字段类型,占位符索引及要替换的值
  • 6. 使⽤ Statement 执⾏ SQL 语句
  • 7. 查询操作:返回结果集 ResultSet,更新操作:返回更新的数量
  • 8. 处理结果集
  • 9. 释放资源
展⽰了通过 JDBC 的 API 向数据库中添加⼀条记录,修改⼀条记录,查询⼀条记录的操作。
从上述的操作流程中可以看出,对于JDBC来说,整个操作非常的繁琐,不但要拼接每一个参数,而且还要按照模板代码的方式,一步步的操作数据库,并且在每次操作完,还要手动关闭连接等,而所有的这些操作步骤都需要在每个方法中重复书写,那有没有一种方法,可以更简单,更方便的操作数据库呢?
答案是肯定的,这就是我们要学习 MyBatis 的真正原因,它可以帮助我们更⽅便、更快速的操作数据库.

🚩什么是MyBatis

MyBatis是一款优秀的 持久层 框架,用于简化JDBC的开发
在上⾯我们提到⼀个词:持久层
持久层:指的就是持久化操作的层, 通常指数据访问层(dao), 是⽤来操作数据库的
简单来说 MyBatis是更简单完成程序和数据库交互的框架,也就是更简单的操作和读取数据库工具,接下来,我们就通过一个入门程序,让大家感受一下通过MyBatis如何来操作数据库

🚩MyBatis⼊⻔

Mybatis操作数据库的步骤:
  • 1. 准备⼯作(创建springboot⼯程、数据库表准备、实体类)
  • 2. 引⼊Mybatis的相关依赖,配置Mybatis(数据库连接信息)
  • 3. 编写SQL语句(注解/XML)
  • 4. 测试

🎈准备工作

📝创建⼯程

创建springboot⼯程,并导⼊ mybatis的起步依赖、mysql的驱动包
Mybatis 是⼀个持久层框架, 具体的数据存储和数据操作还是在MySQL中操作的, 所以需要添加MySQL驱动。

项⽬⼯程创建完成后,⾃动在pom.xml⽂件中,导⼊Mybatis依赖和MySQL驱动依赖 。
<!--Mybatis 依赖包-->
 <dependency>
            <groupId>org.mybatis.spring.boot</groupId>
            <artifactId>mybatis-spring-boot-starter</artifactId>
            <version>3.0.3</version>
 </dependency>

<!--mysql驱动包-->
<dependency>
      <groupId>com.mysql</groupId>
      <artifactId>mysql-connector-j</artifactId>
      <scope>runtime</scope>
</dependency>

📝数据准备

创建⽤⼾表, 并创建对应的实体类User
-- 创建数据库
DROP DATABASE IF EXISTS mybatis_test;
CREATE DATABASE mybatis_test DEFAULT CHARACTER SET utf8mb4;
-- 使⽤数据数据
USE mybatis_test;
-- 创建表[⽤⼾表]
DROP TABLE IF EXISTS userinfo;
CREATE TABLE `userinfo` (
 `id` INT ( 11 ) NOT NULL AUTO_INCREMENT,
 `username` VARCHAR ( 127 ) NOT NULL,
 `password` VARCHAR ( 127 ) NOT NULL,
 `age` TINYINT ( 4 ) NOT NULL,
 `gender` TINYINT ( 4 ) DEFAULT '0' COMMENT '1-男 2-⼥ 0-默认',
 `phone` VARCHAR ( 15 ) DEFAULT NULL,
 `delete_flag` TINYINT ( 4 ) DEFAULT 0 COMMENT '0-正常, 1-删除',
 `create_time` DATETIME DEFAULT now(),
 `update_time` DATETIME DEFAULT now(),
 PRIMARY KEY ( `id` ) 
) ENGINE = INNODB DEFAULT CHARSET = utf8mb4; 
-- 添加⽤⼾信息
INSERT INTO mybatis_test.userinfo ( username, `password`, age, gender, phone )
VALUES ( 'admin', 'admin', 18, 1, '18612340001' );
INSERT INTO mybatis_test.userinfo ( username, `password`, age, gender, phone )
VALUES ( 'zhangsan', 'zhangsan', 18, 1, '18612340002' );
INSERT INTO mybatis_test.userinfo ( username, `password`, age, gender, phone )
VALUES ( 'lisi', 'lisi', 18, 1, '18612340003' );
INSERT INTO mybatis_test.userinfo ( username, `password`, age, gender, phone )
VALUES ( 'wangwu', 'wangwu', 18, 1, '18612340004' );
创建对应的实体类 UserInfo
实体类的属性名与表中的字段名⼀⼀对应
package com.example.mybatis_test.model;

import lombok.Data;

import java.util.Date;

@Data
public class UserInfo {
    private Integer id;
    private String username;
    private String password;
    private Integer age;
    private Integer gender;
    private String phone;
    private Integer deleteFlag;
    private Date createTime;
    private Date updateTime;
}

🎈配置数据库连接字符串

# 数据库连接配置
spring:
  datasource:
    url: jdbc:mysql://127.0.0.1:3306/mybatis_test?characterEncoding=utf8&useSSL=false
    username: root
    password: root
    driver-class-name: com.mysql.cj.jdbc.Driver
注意事项:
如果使⽤ MySQL 是 5.x 之前的使⽤的是"com.mysql.jdbc.Driver",如果是⼤于 5.x 使⽤的
是“com.mysql.cj.jdbc.Driver”.

🎈写持久层代码

在项⽬中, 创建持久层接⼝UserInfoMapper2.

package com.example.mybatis_test.mapper;

import com.example.mybatis_test.model.UserInfo;
import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Select;

import java.util.List;

@Mapper
public interface UserInfoMapper2 {
    //查询所有用户的信息
    @Select("select * from userinfo")
    List<UserInfo>getUserInfoAll();
}
Mybatis的持久层接⼝规范⼀般都叫 XxxMapper   
  • @Mapper注解:表⽰是MyBatis中的Mapper接⼝ 程序运行时,框架会自动生成接口的实现类对象(代理对象),并交给Spring的IOC容器管理.(也就是在程序运行的时候,框架会自动生成UserInfoMapper2接口,然后交给了spring的ioc容器中让spring进行管理,等需要了就注入)
  • @Select注解:代表的就是select查询,也就是注解对应方法的具体实现内容(而select就是执行sql语句,注解里面的内容就是具体实现内容)

🎈单元测试

在创建出来的SpringBoot⼯程中,在src下的test⽬录下,已经⾃动帮我们创建好了测试类 ,我们可以直接使⽤这个测试类来进⾏测试.
package com.example.mybatis_test.mapper;
import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.*;
class UserInfoMapper2Test {
    @Test
    void getUserInfoAll() {

    }
}

此时的测试单元类就自动生成了,再src/test/java/Mapper里面自动生成了测试类。

package com.example.mybatis_test.mapper;

import com.example.mybatis_test.model.UserInfo;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;

import java.util.List;

import static org.junit.jupiter.api.Assertions.*;

@SpringBootTest
class UserInfoMapper2Test {
    @Autowired
    private UserInfoMapper2 userInfoMapper2;
    @Test
    void getUserInfoAll() {
        List<UserInfo>userInfos=userInfoMapper2.getUserInfoAll();
        for (UserInfo userInfo:userInfos) {
            System.out.println(userInfo);
        }
    }
}
记得加 @SpringBootTest 注解, 加载Spring运⾏环境,否则会报错
测试类上添加了注解 @SpringBootTest,该测试类在运⾏时,就会⾃动加载Spring的运⾏环境.
我们通过@Autowired这个注解, 注⼊我们要测试的类, 就可以开始进⾏测试了
返回结果中, 可以看到, 只有SQL语句中查询的列对应的属性才有赋值。sql语句和查询的列属性不对应那么就默认为空值。

🚩MyBatis的基础操作

上⾯我们学习了Mybatis的查询操作, 接下来我们学习MyBatis的增, 删, 改操作 ,在学习这些操作之前, 我们先来学习MyBatis⽇志打印。

🎈打印日志

在Mybatis当中我们可以借助⽇志, 查看到sql语句的执⾏、执⾏传递的参数以及执⾏结果 ,在配置⽂件中进⾏配置即可。
#配置mybatis日志,为了后续观察sql语句的执行
mybatis:
  configuration:
      log-impl: org.apache.ibatis.logging.stdout.StdOutImpl

  • ①: 查询语句  preparing:select*from userinfo  
  • ②: 传递参数及类型   Parameters
  • ③: SQL执⾏结果  cloums以下
我们看到,配置日志文件后,我们就可以观察到,sql语句的执行,以及执行结果,每一行都标注的清清楚楚,并且标注影响的行数。

🎈参数传递

#{}动态参数传递

需求: 查找id=4的⽤⼾,对应的SQL就是: select * from userinfo where id=4
//查询id=4的用户
    @Select("select * from userinfo where id=#{id}")
    UserInfo queryById(Integer id);
如果给where id=4表示只能查找id=4 的数据, 所以SQL语句中的id值不能写成固定数值,需要变为动态的数值 解决⽅案:在queryById⽅法中添加⼀个参数(id), 将⽅法中的参数,传给SQL语句
使⽤ #{} 的⽅式获取⽅法中的参数,使静态数据转成动态数据
如果mapper接⼝⽅法形参只有⼀个普通类型的参数,#{…} ⾥⾯的属性名可以随便写,如:#{id}、# {value}。 建议和参数名保持⼀致
🍭添加测试样例:
  @Test
    void queryById() {
        UserInfo userInfo=userInfoMapper2.queryById(4);
        System.out.println(userInfo);
    }

🍭控制台结果

也可以通过 @Param , 设置参数的别名 , 如果使⽤ @Param 设置别名, #{...}⾥⾯的属性名必须和 @Param 设置的⼀样

🎈增(Insert)

   //增加一个用户
    @Insert("insert into userinfo (username,`password`,age,gender) values" +
            "(#{username},#{password},#{age},#{gender})")
    Integer insert(UserInfo userInfo);

如果语句过长,我们可以直接敲回车,idea会自动给你换行。

🍭添加测试样例

    @Test
    void insert() {
        UserInfo userInfo=new UserInfo();
        userInfo.setUsername("zyf");
        userInfo.setPassword("1005");
        userInfo.setAge(20);
        userInfo.setGender(1);
        Integer count=userInfoMapper2.insert(userInfo);
        System.out.println(count);
    }

🍭控制台结果

如果设置了 @Param 属性, #{...} 需要使⽤ 参数.属性 来获取
  //增加一个用户
    @Insert("insert into userinfo (username,`password`,age,gender) values" +
            "(#{userinfo.username},#{userinfo.password},#{userinfo.age},#{userinfo.gender})")
    Integer insert(@Param("userinfo") UserInfo userInfo);

📝返回主键(实现自增效果)

自增id   @Options(useGeneratedKeys = true, keyProperty = "id")

  • useGeneratedKeys:这会令 MyBatis 使⽤ JDBC 的 getGeneratedKeys ⽅法来取出由数据库内 部⽣成的主键(⽐如:像 MySQL 和 SQL Server 这样的关系型数据库管理系统的⾃动递增字 段),默认值:false.
  • • keyProperty:指定能够唯⼀识别对象的属性,MyBatis 会使⽤ getGeneratedKeys 的返回值或insert 语句的 selectKey ⼦元素设置它的值,默认值:未设置(unset)
Insert 语句默认返回的是 受影响的⾏数
但有些情况下, 数据插⼊之后, 还需要有后续的关联操作, 需要获取到新插⼊数据的id
⽐如订单系统
当我们下完订单之后, 需要通知物流系统, 库存系统, 结算系统等, 这时候就需要拿到订单ID
如果想要拿到⾃增id, 需要在Mapper接⼝的⽅法上添加⼀个Options的注解

 

 注意: 设置 useGeneratedKeys=true 之后, ⽅法返回值依然是受影响的⾏数, ⾃增id 会设置在上 述 keyProperty 指定的属性中.


🎈删(delete)

📝使用注解:

  @Delete("delete from userinfo where id=#{id}")
    Integer delete(Integer id);

📝添加测试样例

此时的表中就删除了id=4的用户。


🎈改(Update)

📝使用注解


    //更新一个用户的姓名
    @Update("update userinfo set username=#{username} where id=#{id}")
    Integer update(String username,Integer id);

📝添加测试样例

   @Test
    void update() {
        Integer count=userInfoMapper2.update("zyf",5);
        System.out.println(count);
    }


🎈查(Select)

我们在上⾯查询时发现, 有⼏个字段是没有赋值的, 只有Java对象属性和数据库字段⼀模⼀样时, 才会进⾏赋值 ,接下来我们多查询⼀些数据
📝使用注解
    @Select("select id, username, `password`, age, gender, phone, delete_flag, create_time, update_time from userinfo")
    List<UserInfo> queryAllUser();

📝添加测试样例

 @Test
    void queryAllUser() {
        List<UserInfo>infoList=userInfoMapper2.queryAllUser();
        for (UserInfo userInfo:infoList) {
            System.out.println(userInfo);
        }
    }

从运⾏结果上可以看到, 我们SQL语句中, 查询了delete_flag, create_time, update_time, 但是这⼏个属性却没有赋值.

原因分析:
当⾃动映射查询结果时, MyBatis 会获取结果中返回的列名并在 Java 类中查找相同名字的属性(忽略⼤⼩写) 。 这意味着如果发现了 ID 列和 id 属性,MyBatis 会将列 ID 的值赋给 id 属性
但是有三个属性和表的列名不一致,就不会进行赋值。

解决办法:
  • 1. 起别名
  • 2. 结果映射
  • 3. 开启驼峰命名

MyBatis 会根据⽅法的返回结果进⾏赋值.
  • ⽅法⽤对象 UserInfo接收返回结果, MySQL 查询出来数据为⼀条, 就会⾃动赋值给对象.
  • ⽅法⽤List<UserInfo>接收返回结果, MySQL 查询出来数据为⼀条或多条时, 也会⾃动赋值给List.
  • 但如果MySQL 查询返回多条, 但是⽅法使⽤UserInfo接收, MyBatis执⾏就会报错

📝起别名

在SQL语句中, 给列名起别名,保持别名和实体类属性名⼀样。  
delete_flag as deleteFlag, create_time as createTime, update_time as updateTime 
🍭使用注解
    @Select("select id, username, `password`, age, gender, phone, delete_flag as deleteFlag, create_time as createTime, update_time as updateTime from userinfo")
    List<UserInfo> queryAllUser2();
SQL语句太⻓时, 使⽤加号 + 进⾏字符串拼接(直接按回车idea自动给你换行)
🍭添加测试
  @Test
    void queryAllUser2() {
        List<UserInfo>infoList=userInfoMapper2.queryAllUser2();
        for (UserInfo userInfo:infoList) {
            System.out.println(userInfo);
        }
    }

📝结果映射

 @Select("select id, username, `password`, age, gender, phone, delete_flag, create_time, update_time from userinfo")
    @Results({
            @Result(column = "delete_flag",property = "deleteFlag"),
            @Result(column = "create_time",property = "createTime"),
            @Result(column = "update_time",property = "updateTime")
    })
    List<UserInfo> queryAllUser();
如果其他SQL, 也希望可以复⽤这个映射关系, 可以给这个Results定义⼀个名,这样我们再queryById方法中我们要想得到id=4的用户,他查询之后的结果也是那三个属性不对应,我们就可以复用下面的@Results注解,用@RestultMap注解定位到@Results注解中id。
使⽤ id 属性给该 Results 定义别名, 使⽤ @ResultMap 注解来复⽤其他定义的 ResultMap
🍭测试结果

📝开启驼峰命名(推荐)

通常数据库列使⽤蛇形命名法进⾏命名(下划线分割各个单词),  Java 属性⼀般遵循驼峰命名法约定.

驼峰命名规则: abc_xyz => abcXyz
  • • 表中字段名:abc_xyz
  • • 类中属性名:abcXyz
为了在这两种命名⽅式之间启⽤⾃动映射,需要将 mapUnderscoreToCamelCase 设置为 true。
mybatis:
 configuration:
  map-underscore-to-camel-case: true #配置驼峰⾃动转换

 🍭配置驼峰自动转换

   //查询所有用户的信息
    @Select("select * from userinfo")
    List<UserInfo>getUserInfoAll();

🍭添加测试样例

    @Test
    void getUserInfoAll() {
        List<UserInfo>userInfos=userInfoMapper2.getUserInfoAll();
        for (UserInfo userInfo:userInfos) {
            System.out.println(userInfo);
        }
    }


相比于起别名和结果映射,开启驼峰命名更加的简单,只需要再配置文件中配置一下,即可。


🚩MyBatis  XML配置文件

上⾯学习了注解的⽅式, 接下来我们学习XML的⽅式
使⽤Mybatis的注解⽅式,主要是来完成⼀些简单的增删改查功能. 如果需要实现复杂的SQL功能,建议使⽤XML来配置映射语句,也就是将SQL语句写在XML配置⽂件中.
MyBatis XML的⽅式需要以下两步:
  • 1. 配置数据库连接字符串和MyBatis
  • 2. 写持久层代码

🎈配置连接字符串和MyBatis

此步骤需要进⾏两项设置,数据库连接字符串设置和 MyBatis 的 XML ⽂件配置。 如果是application.yml⽂件, 配置内容如下:
# 配置 mybatis xml 的⽂件路径,在 resources/mapper 创建所有表的 xml ⽂件
mybatis:
 mapper-locations: classpath:mapper/**Mapper.xml
spring:
  datasource:
    url: jdbc:mysql://127.0.0.1:3306/mybatis_test?characterEncoding=utf8&useSSL=false
    username: root
    password: 105528clzyf.
    driver-class-name: com.mysql.cj.jdbc.Driver
#配置mybatis日志,为了后续观察sql语句的执行
mybatis:
  configuration:
      log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
##配置驼峰⾃动转换
      map-underscore-to-camel-case: true    
# 配置 mybatis xml 的⽂件路径,在 resources/mapper 创建所有表的 xml ⽂件
  mapper-locations: classpath:mapper/UserInfoMapper2.xml

配置mybatis.xml文件,我们需要在src/main/resources/mapper目录下创建xml文件。


🎈写持久层代码

持久层代码分两部分
  • 1. ⽅法定义 Interface
  • 2. ⽅法实现: UserInfoMapper2.xml
Interface用于方法定义,xxx.xml用于方法的实现,实现sql语句。

📝添加 mapper 接⼝

数据持久层的接⼝定义:
package com.example.mybatis_test.mapper;

import com.example.mybatis_test.model.UserInfo;
import org.apache.ibatis.annotations.Mapper;

import java.util.List;

@Mapper
public interface UserInfoXmlMapper2 {
    List<UserInfo>queryAllUser();
}

📝添加 UserInfoXMLMapper2.xml

数据持久成的实现,MyBatis 的固定 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="com.example.mybatis_test.mapper.UserInfoXmlMapper2">
    
</mapper>

创建UserInfoXMLMapper2.xml, 路径参考yml中的配置
# 配置 mybatis xml 的⽂件路径,在 resources/mapper 创建所有表的 xml ⽂件
  mapper-locations: classpath:mapper/UserInfoMapper2.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="com.example.mybatis_test.mapper.UserInfoXmlMapper2">
    <select id="queryAllUser">
        select username,`password`,age,gender,phone, from userinfo
    </select>
</mapper>
以下是对以上标签的说明:
  • <mapper> 标签:需要指定 namespace 属性,表⽰命名空间,值为 mapper 接⼝的全限定名,包括全包名.类名。
  • <select> 查询标签:是⽤来执⾏数据库的查询操作的:
        ◦ id :是和 Interface (接⼝)中定义的⽅法名称⼀样的 ,表⽰对接⼝的具体实现⽅法。
        ◦ resultType :是返回的数据类型,也就是开头我们定义的实体类.

📝单元测试

@SpringBootTest
class UserInfoMapper2Test {
    @Autowired
    private UserInfoMapper2 userInfoMapper2;
    @Test
    void getUserInfoAll() {
        List<UserInfo>userInfos=userInfoMapper2.getUserInfoAll();
        for (UserInfo userInfo:userInfos) {
            System.out.println(userInfo);
        }
    }
}


🎈增删改查操作

📝增(Insert)

🍭UserInfoMapper2接⼝:
 ///增
    Integer insertUser(UserInfo userInfo);
🍭UserInfoMapper.xml实现
 <insert id="insertUser">
        insert into userinfo (username,`password`,age,gender,phone)values (#{username},#{password},#{age},#{gender},#{phone})
    </insert>
如果使⽤@Param设置参数名称的话, 使⽤⽅法和注解类似
🍭UserInfoMapper接⼝:
   ///增
    Integer insertUser(@Param("userinfo") UserInfo userInfo);
🍭UserInfoMapper.xml实现:

    <insert id="insertUser">
        insert into userinfo (username,`password`,age,gender,phone)values (#{userinfo.username},#{userinfo.password},#{userinfo.age},#{userinfo.gender},#{userinfo.phone})
    </insert>
返回⾃增 id
接⼝定义不变, Mapper.xml 实现 设置useGeneratedKeys 和keyProperty属性
<insert id="insertUser" useGeneratedKeys="true" keyProperty="id">
        insert into userinfo (username,`password`,age,gender,phone)values (#{userinfo.username},#{userinfo.password},#{userinfo.age},#{userinfo.gender},#{userinfo.phone})
    </insert>


📝删(Delete)

🍭UserInfoMapper接⼝

  //删
    Integer deleteUser(Integer id);

🍭UserInfoMapper.xml实现:

<delete id="deleteUser">
        delete from userinfo where id=#{id};
    </delete>

🍭测试

  @Test
    void deleteUser() {
        Integer count=userInfoXmlMapper2.deleteUser(2);
        System.out.println(count);
    }


📝改(Update)

🍭UserInfoMapper接⼝
//改
    Integer updateUser(String username,Integer id);

🍭UserInfoMapper.xml实现:

   <update id="updateUser">
        update userinfo set username=#{username} where id=#{id}
    </update>

🍭测试

  @Test
    void updateUser() {
       Integer count=userInfoXmlMapper2.updateUser("zzzzz",7);
        System.out.println(count);
    }


📝查(Select)

同样的, 使⽤XML 的⽅式进⾏查询, 也存在数据封装的问题 ,我们把SQL语句进⾏简单修改, 查询更多的字段内容
🍭UserInfoMapper接⼝
  //查
    List<UserInfo>selectUser();

🍭UserInfoMapper.xml实现:

  <select id="selectUser">
        select id, username,`password`, age, gender, phone, delete_flag,
               create_time, update_time from userinfo
    </select>

🍭测试

 @Test
    void selectUser() {
        List<UserInfo>infoList=userInfoXmlMapper2.selectUser();
        for (UserInfo userInfo:infoList) {
            System.out.println(userInfo);
        }
    }


结果显⽰: deleteFlag, createTime, updateTime 也没有进⾏赋值.
解决办法和注解类似
  • 1. 起别名
  • 2. 结果映射
  • 3. 开启驼峰命名
其中1,3的解决办法和注解⼀样,不再多说, 接下来看下xml如果来写结果映射

🍭结果映射


    <resultMap id="BaseMap" type="com.example.mybatis_test.model.UserInfo">
        <id column="id" property="id"></id>
        <result column="delete_flag" property="deleteFlag"></result>
        <result column="create_time" property="createTime"></result>
        <result column="update_time" property="updateTime"></result>
    </resultMap>
    <select id="selectUser" resultMap="BaseMap">
        select id, username,`password`, age, gender, phone, delete_flag,
               create_time, update_time from userinfo
    </select>


注解方式和XML方式结果映射区别



六一快乐~今天吃了自助餐,几年没有穿裙子今天买了个裙子,给张老师买了白色上袖为it文化节准备上台。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【基本数据结构】平衡二叉树

文章目录 前言平衡二叉树1 简介2 旋转2.1 左旋2.2 右旋2.3 何时旋转 3 插入节点4 删除节点5 代码 参考资料写在最后 前言 本系列专注更新基本数据结构&#xff0c;现有以下文章&#xff1a; 【算法与数据结构】数组. 【算法与数据结构】链表. 【算法与数据结构】哈希表. 【…

33【Aseprite 作图】树——拆解

1 树叶 画树叶真累啊&#xff0c;可以先画一个轮廓&#xff0c;细节一点点修 2 1 2 &#xff1b;2 2 2 &#xff08;横着横&#xff09;&#xff0c;这样一点点画树叶 填充颜色&#xff0c;用了喷雾工具 2 树干部分 轮廓部分&#xff0c;左边的是3 3 3 &#xff1b;上下都是…

从不同角度看如何让大模型变得更聪明呢?

算法创新&#xff0c;从代码上优化大模型&#xff0c;可以采取一系列策略来提升其性能和效率。 算法优化&#xff1a;对模型的算法进行精细调整&#xff0c;如改进神经网络架构&#xff0c;使用更高效的层&#xff08;如深度可分离卷积&#xff09;&#xff0c;或者优化递归神经…

前端地图中,已知一个点位,获取相同经度或者纬度下的,某个距离的另一个点位

效果图说明&#xff1a;我在圆的中心点位&#xff0c;找到他某个直线距离的另个一点&#xff0c;标注两者之间的距离。如图所示是25000米。 沿纬度方向移动 在相同经度下&#xff0c;计算沿纬度方向移动1000米的新点位&#xff1a; function calculateLatitudePoint(lat, ln…

回归预测 | MATLAB实现基于GOOSE-LightGBM的多特征输入单输出数据回归预测(鹅优化算法)

回归预测 | MATLAB实现基于GOOSE-LightGBM的多特征输入单输出数据回归预测(鹅优化算法) 目录 回归预测 | MATLAB实现基于GOOSE-LightGBM的多特征输入单输出数据回归预测(鹅优化算法)效果一览基本介绍程序设计参考资料 效果一览 基本介绍 MATLAB实现基于LightGBM算法的数据回归预…

Qt第三方库QicsTable简单实例(1)

闲来无事&#xff0c;无意间看到一个Qics表格操作第三方库&#xff0c;自己写了一个特别简单的实例&#xff0c;效果如图所示&#xff1a; 操作界面的数据还是特别快的&#xff0c;因为使用了模型

java并发处理机制

在Java中&#xff0c;并发处理机制主要是通过线程来实现的。Java提供了丰富的类和接口来支持多线程编程&#xff0c;主要集中在 java.util.concurrent 包中。以下是一些关键的并发处理机制&#xff1a; 1.线程创建&#xff1a;可以通过继承 Thread 类或实现 Runnable 接口来创建…

前端Vue小兔鲜儿电商项目实战Day06

一、本地购物车 - 列表购物车 1. 基础内容渲染 ①准备模板 - src/views/cartList/index.vue <script setup> const cartList [] </script><template><div class"xtx-cart-page"><div class"container m-top-20"><div…

C语言:如何写文档注释、内嵌注释、行块注释?

技术答疑流程 扫描二维码&#xff0c;添加个人微信&#xff1b;支付一半费用&#xff0c;获取答案&#xff1b;如果满意&#xff0c;则支付另一半费用&#xff1b; 知识点费用&#xff1a;10元 项目费用&#xff1a;如果有项目任务外包需求&#xff0c;可以微信私聊

Wpf 使用 Prism 实战开发Day31

登录数据绑定 1.首先在LoginViewModel 登录逻辑处理类中&#xff0c;创建登录要绑定属性和命令 public class LoginViewModel : BindableBase, IDialogAware {public LoginViewModel(){ExecuteCommand new DelegateCommand<string>(Execure);}public string Title { ge…

Arduino烧录esp8266

default_encoding: cp936 Assume aggressive ‘core.a’ caching enabled. Note: optional global include file ‘arduino_modified_sketch_764314\Blink.ino.globals.h’ does not exist. Read more at https://arduino-esp8266.readthedocs.io/en/latest/faq/a06-global-bui…

数据管理知识体系必知的14张语境关系图

近期对数据管理知识体系中的语境关系图进行了整体学习梳理,总共有14张图,具体如下,供大家参考。应该说语境关系图和环境因素六边形图是各有侧重、互为补充关系。语境关系图是环境因素六边形图的细化,描述了每个知识领域中的细节,相当于数据管理的微观视角, 包括与人员、 …

秒杀基本功能开发(显示商品列表和商品详情)

文章目录 1.数据库表设计1.商品表2.秒杀商品表3.修改一下秒杀时间为今天到明天 2.pojo和vo编写1.com/sxs/seckill/pojo/Goods.java2.com/sxs/seckill/pojo/SeckillGoods.java3.com/sxs/seckill/vo/GoodsVo.java 3.Mapper编写1.GoodsMapper.java2.GoodsMapper.xml3.分别编写Seck…

JCR一区级 | Matlab实现TCN-LSTM-MATT时间卷积长短期记忆神经网络多特征分类预测

JCR一区级 | Matlab实现TCN-LSTM-MATT时间卷积长短期记忆神经网络多特征分类预测 目录 JCR一区级 | Matlab实现TCN-LSTM-MATT时间卷积长短期记忆神经网络多特征分类预测分类效果基本介绍程序设计参考资料 分类效果 基本介绍 1.JCR一区级 | Matlab实现TCN-LSTM-MATT时间卷积长短…

用户画像知识点补充——多数据源

引入 针对用户画像项目来说&#xff08;产品&#xff09;必须要支持从多种数据源加载业务数据&#xff0c;构建用户标签。 在之前的标签模型开发中&#xff0c;主要是为了简化开发复杂度&#xff0c;业务数据统一存储到HBase表中。 数据源包含如下几个方面&#xff1a; 存储H…

民国漫画杂志《时代漫画》第38期.PDF

时代漫画38.PDF: https://url03.ctfile.com/f/1779803-1248636380-dd7daa?p9586 (访问密码: 9586) 《时代漫画》的杂志在1934年诞生了&#xff0c;截止1937年6月战争来临被迫停刊共发行了39期。 ps: 资源来源网络!

R19 NR移动性增强概况

随着5G/5G-A技术不断发展和业务需求的持续增强&#xff0c;未来网络的部署将不断向高频演进。高频小区的覆盖范围小&#xff0c;用户将面临更为频繁的小区选择、重选、切换等移动性过程。 为了提升网络移动性能和保障用户体验&#xff0c;移动性增强一直是3GPP的热点课题。从NR…

11.1 排序算法

目录 11.1 排序算法 11.1.1 评价维度 11.1.2 理想排序算法 11.1 排序算法 排序算法&#xff08;sorting algorithm&#xff09;用于对一组数据按照特定顺序进行排列。排序算法有着广泛的应用&#xff0c;因为有序数据通常能够被更高效地查找、分析和处理。 如图 1…

修改element-ui el-radio颜色

修改element-ui el-radio颜色 需求效果图代码实现 小结 需求 撤销扣分是绿色&#xff0c;驳回是红色 效果图 代码实现 dom <el-table-columnlabel"操作"width"200px"><template v-slot"scope"><el-radio-group v-model"s…

短剧源码系统深层次解析:技术架构与实现

短剧源码系统作为短视频内容生产与分发的核心技术&#xff0c;其技术实现对于开发者和运营者至关重要。本文将深入探讨短剧源码系统的关键技术架构&#xff0c;特别是前端框架uni-app和Vue&#xff0c;以及后端框架ThinkPHP5和Workerman的应用。 前端框架&#xff1a;uni-app与…