MyBatis:核心概念 + 环境搭建 + CRUD

一、简介

本文适合 Java 开发初学者,从零梳理 MyBatis 核心知识,包含环境搭建、配置详解、CRUD 实战、常见避坑,可直接用于学习、复习与项目开发。

二、MyBatis 基础概念与环境搭建

1. JDBC 痛点与 MyBatis 优势

在学习 MyBatis 之前,我们先回顾传统JDBC 开发的缺陷,这也是 MyBatis 诞生的核心原因。

JDBC 操作缺陷
  1. 资源频繁创建与释放,严重影响性能每次操作数据库都需要创建 Connection、Statement,用完直接关闭,频繁 IO 造成资源浪费。
  2. SQL 硬编码,维护成本极高SQL 语句直接写在 Java 代码中,修改 SQL 必须重新编译、打包、部署,无法灵活配置。
  3. 结果集解析繁琐,手动封装冗余 查询返回 ResultSet 后,需要手动getXxx()取值、手动 set 到实体类,代码量大且易出错。
  4. 代码冗余,开发效率低下连接、参数设置、结果解析大量重复代码,不符合开发规范。

典型 JDBC 冗余代码:

复制代码
// 1. 加载驱动、建立连接
Class.forName("com.mysql.jdbc.Driver");
Connection conn = DriverManager.getConnection(url, user, pwd);
// 2. 编写SQL(硬编码)
String sql = "select * from user where id = ?";
PreparedStatement pstmt = conn.prepareStatement(sql);
// 3. 设置参数
pstmt.setInt(1, id);
// 4. 执行查询
ResultSet rs = pstmt.executeQuery();
// 5. 手动封装结果集
User user = null;
if(rs.next()){
    user = new User();
    user.setId(rs.getInt("id"));
    user.setUsername(rs.getString("username"));
}
MyBatis 优势与解决方案

MyBatis 是一款优秀的持久层框架,底层依然基于 JDBC,但做了高度封装:

  1. 连接池管理:避免频繁创建 / 释放连接,大幅提升性能;
  2. SQL 与 Java 代码解耦:SQL 统一写在 XML 配置文件中,修改无需编译 Java;
  3. 自动结果集映射:自动将 ResultSet 封装为实体对象,无需手动解析;
  4. 参数自动映射 :支持简单类型、实体类参数,直接使用#{}占位符;
  5. 轻量级、学习成本低:相比 Hibernate,MyBatis 更灵活、可控性更强。

2. 项目环境配置

我们以Maven 项目为例,搭建标准 MyBatis 开发环境。

(1)Maven 依赖导入

pom.xml中导入核心依赖:MyBatis 核心包、MySQL 驱动、日志包、Junit 测试

复制代码
<dependencies>
    <!-- MyBatis核心依赖 -->
    <dependency>
        <groupId>org.mybatis</groupId>
        <artifactId>mybatis</artifactId>
        <version>3.5.9</version>
    </dependency>
    <!-- MySQL驱动 -->
    <dependency>
        <groupId>mysql</groupId>
        <artifactId>mysql-connector-java</artifactId>
        <version>5.1.49</version>
    </dependency>
    <!-- 日志包 -->
    <dependency>
        <groupId>log4j</groupId>
        <artifactId>log4j</artifactId>
        <version>1.2.17</version>
    </dependency>
    <!-- 测试包 -->
    <dependency>
        <groupId>junit</groupId>
        <artifactId>junit</artifactId>
        <version>4.13.2</version>
        <scope>test</scope>
    </dependency>
</dependencies>
(2)MyBatis 核心配置文件(sqlMapConfig.xml)

resources目录下创建核心配置文件,配置数据源、事务、加载映射文件

复制代码
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE configuration
        PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
    <!-- 环境配置:默认使用development -->
    <environments default="development">
        <environment id="development">
            <!-- 事务管理:JDBC事务 -->
            <transactionManager type="JDBC"/>
            <!-- 数据源:POOLED连接池 -->
            <dataSource type="POOLED">
                <property name="driver" value="com.mysql.jdbc.Driver"/>
                <property name="url" value="jdbc:mysql://localhost:3306/mybatis_demo?serverTimezone=Asia/Shanghai&amp;useSSL=false&amp;characterEncoding=utf8"/>
                <property name="username" value="root"/>
                <property name="password" value="abc123"/>
            </dataSource>
        </environment>
    </environments>

    <!-- 加载Mapper映射文件 -->
    <mappers>
        <mapper resource="mapper/UserMapper.xml"/>
    </mappers>
</configuration>
(3)项目标准结构
复制代码
src/main/java
  com.qcby
    ├── dao/          DAO接口层
    ├── entity/       实体类
src/main/resources
  ├── mapper/         Mapper XML映射文件
  ├── sqlMapConfig.xml  MyBatis核心配置

三、核心配置与映射关系

1. 实体类与 Mapper 映射

MyBatis 实现自动封装的前提:数据库表 ↔ 实体类 严格映射

(1)实体类编写规范
  1. 类属性名 与 数据库表字段名完全一致
  2. 必须提供无参构造
  3. 必须生成getter/setter
  4. 必须重写toString() 方便测试打印。

User 实体类示例:

复制代码
public class User {
    private Integer id;
    private String username;
    private String sex;
    private Date birthday;
    private String address;

    // 无参构造
    public User(){}
    // getter/setter
    // toString()
}
(2)DAO 接口与 XML 绑定规则

MyBatis 采用接口代理模式开发,无需编写实现类:

  1. 创建 DAO 接口(如UserDao.java);
  2. XML 文件通过namespace绑定接口全类名
  3. XML 中 SQL 标签的id 必须与 接口方法名完全一致
  4. parameterType/resultType 与方法参数、返回值匹配。

接口示例:

复制代码
public interface UserDao {
    List<User> findAll();
    User findUserById(Integer id);
}

XML 绑定示例:

复制代码
<!-- namespace:绑定接口全限定名 -->
<mapper namespace="com.qcby.dao.UserDao">
    <!-- id:与接口方法名一致 -->
    <select id="findAll" resultType="com.qcby.entity.User">
        select * from user order by id desc
    </select>
</mapper>

2. 参数与结果映射

(1)参数传递(parameterType)

用于指定 SQL 入参类型,支持:

  • 基本类型:IntegerString
  • 实体类型:com.qcby.entity.User
  • 取值语法:#{属性名} 安全占位符,防 SQL 注入

xml

复制代码
<select id="findUserById" parameterType="Integer" resultType="com.qcby.entity.User">
    select * from user where id = #{id}
</select>
(2)结果集封装(resultType)

指定 SQL 返回值类型:

  • 查询单条数据:直接写实体类全类名
  • 查询多条数据 :依然写实体类,MyBatis 自动封装为List<实体>

示例:查询所有用户xml

复制代码
<select id="findAll" resultType="com.qcby.entity.User">
    select * from user
</select>

四、增删改查(CRUD)实战

1. 查询操作(Select)

查询是 MyBatis 最常用操作,分为全量查询、精确查询、模糊查询

(1)全量查询 findAll
复制代码
<select id="findAll" resultType="com.qcby.entity.User">
    select * from user order by id desc;
</select>

接口方法:

复制代码
List<User> findAll();
(2)条件查询:根据 ID 查询(单条结果)
复制代码
<select id="findUserById" parameterType="Integer" resultType="com.qcby.entity.User">
    select * from user where id = #{id};
</select>

接口方法:

复制代码
User findUserById(Integer id);
(3)模糊查询:根据用户名模糊匹配

重点:like 必须使用 concat 拼接,避免 SQL 注入

xml

复制代码
<select id="findUserByUserName" parameterType="String" resultType="com.qcby.entity.User">
    select * from user where username like concat('%', #{username}, '%');
</select>

接口方法:

复制代码
List<User> findUserByUserName(String username);
(4)字段映射异常处理

常见问题 :查询结果部分字段为null根本原因 :实体类属性名 ≠ 数据库列名解决方案

  1. 统一名称(推荐)
  2. 使用resultMap手动映射

2. 增删改操作(Insert/Update/Delete)

增删改属于写操作 ,必须注意:必须手动提交事务 session.commit()

(1)插入数据 insert
复制代码
<insert id="insert" parameterType="com.qcby.entity.User">
    insert into user(username,sex,birthday,address)
    values(#{username},#{sex},#{birthday},#{address});
</insert>

接口方法:

复制代码
int insert(User user);
(2)修改数据 update

必须传入包含 id的实体对象

复制代码
<update id="update" parameterType="com.qcby.entity.User">
    update user 
    set username=#{username},sex=#{sex},birthday=#{birthday},address=#{address}
    where id=#{id};
</update>
(3)删除数据 delete
复制代码
<delete id="delete" parameterType="Integer">
    delete from user where id=#{id};
</delete>
(4)事务提交机制

查询不需要提交事务;增删改必须提交!

复制代码
@Test
public void addUser(){
    User user = new User();
    user.setUsername("张三");
    user.setSex("男");
    // 执行插入
    userDao.insert(user);
    // 必须提交事务
    session.commit();
}

五、测试工具类(通用版)

为了简化测试,可封装 MyBatis 工具类:

复制代码
public class MyBatisUtil {
    private static SqlSessionFactory sqlSessionFactory;
    static{
        try {
            String resource = "sqlMapConfig.xml";
            InputStream inputStream = Resources.getResourceAsStream(resource);
            sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
    // 获取SqlSession
    public static SqlSession getSession(){
        return sqlSessionFactory.openSession(true); // 自动提交事务
    }
}
相关推荐
DN金猿6 小时前
spring.cloud.nacos.discovery.server-addr和spring.cloud.nacos.server-addr区别
java·开发语言·nacos·springcloud·sca
ChampaignWolf7 小时前
在 Eclipse 中使用 Tabnine
java·ide·eclipse
JAVA面经实录9177 小时前
Java多线程并发高频面试100题(完整版·含答案·背诵版)
java·开发语言·面试
XiYang-DING8 小时前
【Java EE】TCP—流量控制和拥塞控制
java·tcp/ip·java-ee
BIG_PEI8 小时前
检查并安装Redis
java
大貔貅喝啤酒8 小时前
基于Windows下载安装Android Studio 3.3.2版本教程(2026详细图文版)
android·java·windows·android studio
XS0301068 小时前
MyBatis基础实战笔记一
笔记·mybatis
奋斗的小方8 小时前
Java基础篇09:项目实战
java·开发语言
海兰8 小时前
【第21篇-续】graph-Stream-Node改造为适配openAI模型示例
java·人工智能·spring boot·spring·spring ai