0
点赞
收藏
分享

微信扫一扫

MyBatis-Plus,可快速了解mybatis-plus,快速熟悉API方法,包含了基本增删改查,分页查询,高级查询,排序,分组等

未定义变量 2022-04-13 阅读 54
java

MyBatis-Plus

一.什么是MyBatis_Plus

MyBatis-Plus(简称 MP)是一个 MyBatis 的增强工具,在 MyBatis 的基础上只做增强不做改变,为简化开发、提高效率而生。

特点:无入侵,损耗小,支持Lambda形式调用,支持主键自动生成,支持ActiveRecordd模式,支持自定义全局通用操作,内置代码生成器,内置分页插件,分页检查支持多种数据库,内置性能分析插件,内置全局拦截插件,

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-CWGGI8mH-1649552885085)(E:\java预习资料\项目二\总结图\MyBatis-puls\Snipaste_2022-04-09_20-53-55.png)]

二.所需配置

pom依赖

<parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.4.3</version>
    </parent>
    <dependencies>
        <dependency>
            <groupId>com.baomidou</groupId>
            <artifactId>mybatis-plus-boot-starter</artifactId>
            <version>3.4.0</version>
        </dependency>
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>druid-spring-boot-starter</artifactId>
            <version>1.1.17</version>
        </dependency>
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>8.0.22</version>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.18.16</version>
            <scope>provided</scope>
        </dependency>
    </dependencies>

启动类配置

//分页拦截器
    @Bean
    public MybatisPlusInterceptor mybatisPlusInterceptor() {
        MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
        PaginationInnerInterceptor paginationInnerInterceptor = new PaginationInnerInterceptor(DbType.MYSQL);
        paginationInnerInterceptor.setOverflow(true); //合理化
        interceptor.addInnerInterceptor(paginationInnerInterceptor);
        return interceptor;
    }

properties配置

#mysql
spring.datasource.url=jdbc:mysql://localhost:3306/mybatis-plus?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8&nullCatalogMeansCurrent=true
spring.datasource.username=root
spring.datasource.password=admin
spring.datasource.driver-class-name=com.mysql.jdbc.Driver
# 配置slq打印日志
logging.level.cn.wolfcode.mp.mapper=debug

三.类和重要注解

建domain和mapper和测试类

Employee

@ToString
@Getter
@Setter
@TableName("t_employee")//指定当前实体类映射的表名
public class Employee {
    @TableId(type = IdType.AUTO)//id自增
    private Long id;//默认生成的id策略是雪花算法
    //@TableField("uname")//明确指定当前映射的列
    private String name;
    private String password;
    private String email;
    private int age;
    private int admin;
    //@TableField(exist = false)//明确指定当前字段不进行映射
    private Long deptId;
    //@TableField(exist = false)
    private Department department;
}

EmployeeMapper

public interface EmployeeMapper extends BaseMapper<Employee> {
}

就可以使用了,mybatis-plus内部提供了17个方法来使用.不要写xml和sql语句

四.测试实践

使用17个方法练习

@SpringBootTest(classes = App.class)
public class CRUDTest {
    @Autowired
    private EmployeeMapper employeeMapper;
}

1.修改的方法四个

 /**
     * Mybatis-plus sql 拼接规则
     * 1>在sql拼接时,如果传入的参数是实体类对象,如果对象属性值为null,该属性不参与sql拼接
     * 2>在sql拼接时,如果传入的参数是实体类对象,,如果对象属性是基本类型.
     * 有默认值,myBatis-plus认为有值,该属性参与sql拼接
     *
     * 解决方案呢?
     * 1>使用基本类型的包装类型
     * 2>按照先查询,再替换,后更新顺序执行逻辑
     * 3>使用mapper接口中 update(null ,wrapper) 方法 [局部更新(部分字段)]
     */
//Preparing: UPDATE t_employee SET uname=?, age=?, admin=? WHERE id=?
    //zhiyi(String), 0(Integer), 0(Integer), 1(Long)
    @Test
    public void TestUpdate(){
        Employee employee = new Employee();
        employee.setId(1L);
        employee.setName("zhiyi");
        //返回受影响行数
        int i = employeeMapper.updateById(employee);
    }
    @Test
    public void TestUpdate1(){
        //条件构造器--暂时理解where操作
        UpdateWrapper<Employee> wrapper = new UpdateWrapper<>();
        //eq是等于的意思,
        wrapper.eq("id",1L);
        wrapper.set("name","zhiyi2");
        //null的意思不传实体类,直接set进去sql里
        employeeMapper.update(null,wrapper);
    }
    /**
     * 总结:updateById用于,知道id并更新所有字段
     * update,加上wrapper条件更新部分字段
     */

2.删除

//删除
    @Test
    public void TestDelete(){
        //DELETE FROM t_employee WHERE id=?
        employeeMapper.deleteById(1L);
    }

    @Test
    public void TestDeleteBatchIds(){
        //需求:删除(根据ID 批量删除)
        //Preparing: DELETE FROM t_employee WHERE id IN ( ? , ? , ? )
        //Parameters: 1(Long), 2(Long), 3(Long)
        employeeMapper.deleteBatchIds(Arrays.asList(1L,2L,3L));
    }

    @Test
    public void TestDeleteMap(){
        //需求:删除name=zhiyi并且age=18的员工信息
        //key:条件列  value:条件值
        HashMap<String, Object> map = new HashMap<>();
        map.put("name","zhiyi");
        map.put("age",18);
        employeeMapper.deleteByMap(map);
        //DELETE FROM t_employee WHERE name = ? AND age = ?
        //zhiyi(String), 18(Integer)
    }

    @Test
    public void TestDelete1(){
        //需求:删除name=zhiyi并且age=18的员工信息
        //查询相关条件建议用queryWrapper
        QueryWrapper<Employee> wrapper = new QueryWrapper<>();
        wrapper.eq("name","zhiyi");
        wrapper.eq("age",18);
        employeeMapper.delete(wrapper);
        //DELETE FROM t_employee WHERE (name = ? AND age = ?)
        //zhiyi(String), 18(Integer)
    }

3.查询

//查询
    @Test
    public void TestSelectById(){
        //需求:查询id=4的员工信息
        System.out.println(employeeMapper.selectById(4L));
        //SELECT id,name,password,email,age,admin,dept_id FROM t_employee WHERE id=?
    }

    @Test
    public void TestSelectBatchIds(){
        //需求:查询id=4,id=5的员工信息
        List<Employee> employees = employeeMapper.selectBatchIds(Arrays.asList(4L, 5L));
        //SELECT id,name,password,email,age,admin,dept_id FROM t_employee WHERE id IN ( ? , ? )
    }

    @Test
    public void TestSelectByMap(){
        //需求: 查询name=zhiyi, age=19的员工信息
        HashMap<String, Object> map = new HashMap<>();
        map.put("name","zhiyi");
        map.put("age",19);
        List<Employee> employees = employeeMapper.selectByMap(map);
        //SELECT id,name,password,email,age,admin,dept_id FROM t_employee WHERE name = ? AND age = ?
    }

    @Test
    public void TestSelectCount(){
        //需求: 查询满足条件的所有的员工个数
        //wrapper这里可以理解成where
        QueryWrapper<Employee> wrapper = new QueryWrapper<>();
        //没有加就默认查询全部
        //employeeMapper.selectCount(null);
        Integer count = employeeMapper.selectCount(wrapper);
        //Preparing: SELECT COUNT( 1 ) FROM t_employee
    }

    @Test
    public void TestSelectList(){
        //需求: 查询满足条件的所有的员工信息, 返回List<Employee>
        QueryWrapper<Employee> wrapper = new QueryWrapper<>();
        List<Employee> list = employeeMapper.selectList(wrapper);
        list.forEach(System.err::println);
        //SELECT id,name,password,email,age,admin,dept_id FROM t_employee
    }

    @Test
    public void TestSelectMaps(){
        //需求: 查询满足条件的所有的员工信息,
        // 返回List<Map<String, Object>>  底层将每条数据封装成HashMap
        QueryWrapper<Employee> wrapper = new QueryWrapper<>();
        List<Map<String, Object>> list = employeeMapper.selectMaps(wrapper);
        list.forEach(System.err::println);
        //SELECT id,name,password,email,age,admin,dept_id FROM t_employee
    }

    /**
     * 什么时候用用selectList查询 ,什么时候用selectMap查询
     * 执行的sql返回的列能封装成对象,使用selectList,如果不能,使用selectMap   目的:防止封装的数据丢失
     * eg:SELECT dept_id ,count(id) FROM t_employee group by dept_id
     * 执行之后,返回count 列是无法封装到employee对象中,但是可以封装到map中
     */

4.分页查询

@Test
    public void TestSelectPage() {
        QueryWrapper<Employee> wrapper = new QueryWrapper<>();
        //需求:查询第二页员工数据, 每页显示3条, (分页返回的数据是实体对象)
        //步骤1.当前页  参数2每页显示条数
        Page<Employee> page = new Page<>(2, 3);
        //步骤2.编写分页代码
        Page<Employee> page1 = employeeMapper.selectPage(page, wrapper);
        System.out.println(page==page1);
        System.out.println("当前页:" + page.getCurrent());
        System.out.println("每页显示条数:" + page.getSize());
        System.out.println("总页数:" + page.getPages());
        System.out.println("总数:" + page.getTotal());
        System.out.println("当前页数据:" + page.getRecords());
    }

五.条件构造器的修改操作和查询操作

1.修改操作

@Test
    public  void testUpdate(){
        //条件构造器---暂时理解where操作
        //需求:将id=4的用户name改为zhiyi
        UpdateWrapper<Employee> wrapper = new UpdateWrapper<>();
        wrapper.eq("id",4L);
        wrapper.set("name","zhiyi");
        employeeMapper.update(null,wrapper);
        //UPDATE t_employee SET name=? WHERE (id = ?)
    }

    @Test
    public  void testUpdate2(){
        //条件构造器---暂时理解where操作
        //需求:将id=4的用户name改为zhiyi
        UpdateWrapper<Employee> wrapper = new UpdateWrapper<>();
        wrapper.eq("id",4L);
        wrapper.setSql("name='zhiyi'");//sql片段,直接插进去sql中
        //Preparing: UPDATE t_employee SET name=? WHERE (id = ?)
    }

    @Test
    public  void testUpdate3(){
        //条件构造器---暂时理解where操作
        //需求:将id=4的用户name改为zhiyi
        LambdaUpdateWrapper<Employee> wrapper = new LambdaUpdateWrapper<>();
        wrapper.eq(Employee::getId,4L);
        wrapper.set(Employee::getName,"zhiyi");
        employeeMapper.update(null,wrapper);
    }

2…查询操作

//查询
    @Test
    public  void testQuery1(){
        //需求:查询name=zhiyi, age=19的用户
        QueryWrapper<Employee> wrapper = new QueryWrapper<>();
        wrapper.eq("name","zhiyi");
        wrapper.eq("age",19);
        System.out.println(employeeMapper.selectList(wrapper));
        //SELECT id,name,password,email,age,admin,dept_id FROM t_employee WHERE (name = ? AND age = ?)
    }

    @Test
    public  void testQuery2(){
        //需求:查询name=zhiyi, age=19的用户
        LambdaQueryWrapper<Employee> wrapper = new LambdaQueryWrapper<>();
        wrapper.eq(Employee::getName,"zhiyi");
        wrapper.eq(Employee::getAge,19);
        System.out.println(employeeMapper.selectList(wrapper));
        //SELECT id,name,password,email,age,admin,dept_id FROM t_employee WHERE (name = ? AND age = ?)
    }

六.高级查询

1.包含了模糊查询,分页查询,排序,分组

public void testQuery1(){
        QueryWrapper<Employee> wrapper = new QueryWrapper<>();
        //wrapper.select("name","age");
        wrapper.select("name,age");//sql片段
        List<Employee> list = employeeMapper.selectList(wrapper);
        list.forEach(System.out::println);
        //Preparing: SELECT name,age FROM t_employee
    }

    @Test
    public void testQuery2(){
        //需求:查询所有员工, 返回员工以a字母开头的列
        QueryWrapper<Employee> wrapper = new QueryWrapper<>();
        wrapper.select(Employee.class,tableFieldInfo -> tableFieldInfo.getProperty().startsWith("a"));
        employeeMapper.selectList(wrapper);
    }

    @Test
    public void testQuery3(){
        //需求:查询所有员工信息按age正序排, 如果age一样, 按id正序排
        QueryWrapper<Employee> wrapper = new QueryWrapper<>();
        //wrapper.orderByAsc("age");
        //wrapper.orderByDesc("id");//FROM t_employee ORDER BY age ASC,id DESC

        //开关方法
        //参数1开关参数.true/false决定是否执行后续逻辑(逻辑) 参数2 排序列
        //wrapper.orderByAsc(true,"age");//ORDER BY age ASC

        //参数1:开关参数,  参数2.是否正序,参数:排序列
        wrapper.orderBy(true,true,"age");//ORDER BY age ASC
        employeeMapper.selectList(wrapper);
    }
    @Test
    public void testQuery4(){
        //需求: 以部门id进行分组查询,查每个部门员工个数
        QueryWrapper<Employee> wrapper = new QueryWrapper<>();
        wrapper.select("dept_id,count(id) count");
        wrapper.groupBy("dept_id");
        List<Map<String, Object>> maps = employeeMapper.selectMaps(wrapper);
        //Preparing: SELECT dept_id,count(id) count FROM t_employee GROUP BY dept_id
    }

    @Test
    public void testQuery5(){
        //需求: 以部门id进行分组查询,查每个部门员工个数, 将大于3人的部门过滤出来
        QueryWrapper<Employee> wrapper = new QueryWrapper<>();
        wrapper.select("dept_id,count(id) count");
        wrapper.groupBy("dept_id");
        //wrapper.having("count>3");//GROUP BY dept_id HAVING count>3
        wrapper.having("count>{0}",3);//GROUP BY dept_id HAVING count>?
        List<Map<String, Object>> maps = employeeMapper.selectMaps(wrapper);
    }

    @Test
    public void testQuery6(){
        //需求:查询name=zhiyi员工信息
        QueryWrapper<Employee> wrapper = new QueryWrapper<>();
        HashMap<String, Object> map = new HashMap<>();
        map.put("name","zhiyi");
        map.put("age",null);
        wrapper.allEq(map);
        wrapper.allEq(map,false);
        employeeMapper.selectList(wrapper);//WHERE (name = ? AND age IS NULL AND name = ?)
    }

    @Test
    public void testQuery7(){
        QueryWrapper<Employee> wrapper = new QueryWrapper<>();
        //wrapper.in("id",1L,2L,3L);//WHERE (id IN (?,?,?))
        wrapper.inSql("id","1,2,3");// WHERE (id IN (1,2,3))
        employeeMapper.selectList(wrapper);
    }

2.模糊查询

//模糊查询
    @Test
    public void testQuery8(){
        QueryWrapper<Employee> wrapper = new QueryWrapper<>();
        //wrapper.like("name","yi");// %yi%
        //wrapper.likeRight("name","yi");// yi%
        wrapper.likeLeft("name","yi");//%yi
        employeeMapper.selectList(wrapper);
    }


    @Test
    public void testQuery9() {
        //需求:查询age=19  或者 name=zhiyi  或者id=4的用户
        QueryWrapper<Employee> wrapper = new QueryWrapper<>();
        wrapper.eq("age",19)
                .or()
                .eq("name","zhiyi")
                .or()
                .eq("id",4L);
        employeeMapper.selectList(wrapper);
        //WHERE (age = ? OR name = ? OR id = ?)
    }


    @Test
    public void testQuery10() {
        //需求:查询name含有yi的字样,并且小于18或者大于30的用户
        QueryWrapper<Employee> wrapper = new QueryWrapper<>();
        //wrapper.like("name","yi")
          //      .lt("age",18).or().gt("age",30);
        //WHERE (name LIKE ? AND age < ? OR age > ?)


        wrapper.like("name","yi")
                .and(wrapper2->wrapper2.lt("age",19).or().gt("age",30));
        // WHERE (name LIKE ? AND (age < ? OR age > ?))
        employeeMapper.selectList(wrapper);
    }

3.自定义sql

自定义sql和平时的mybatis一样

/自定义sql
    @Test
    public void testQuery11() {
        List<Employee> list = employeeMapper.listByXmlSingle();
    }

    @Test
    public void testQuery12() {
        List<Employee> list = employeeMapper.listByXmlJoin();
        list.forEach(System.err::println);
    }

EmployeeMapper中

public interface EmployeeMapper extends BaseMapper<Employee> {
    List<Employee> listByXmlSingle();
    List<Employee> listByXmlJoin();
}

EmployeeMapper.xml

<select id="listByXmlSingle" resultMap="BaseResultMap">
        select id,name,password,email,age,admin,dept_id
        from t_employee
    </select>
    <select id="listByXmlJoin" resultMap="BaseResultMap">
        select e.id,e.name,e.password,e.email,e.age,e.admin,e.dept_id
        ,d.id d_id,d.name d_name, d.sn d_sn
        from t_employee e left join department d on e.dept_id = d.id
    </select>

就这么多

七.通用的Service接口

1:自定义服务接口集成IService接口

/**
 * 自定义服务层接口
 * 1>自定义接口继承IService父接口
 * 2>明确指定一个泛型,当前接口操作的实体类对象:Employee
 */
public interface IEmployeeService extends IService<Employee> {
    IPage<Employee> query(EmployeeQuery qo);
}

2:服务接口实现类集成IService接口实现类ServiceImpl同时实现自定义接口

注意ServiceImpl实现类泛型:

泛型1:实体类的mapper接口

泛型2:实体类

/**
 * 自定义访问层接口实现类写法
 * 1.>实现自定义访问接口IEmployeeService
 * 2>继承ServiceImpl通用接口实现类
 * 3>明确指出两个泛型
 *  1.当前操作的实体类对象对应Mapper接口EmployeeMapper
 *  2.当前操作的实体类对象Employee
 */
@Service
public class EmployeeServiceImpl extends ServiceImpl<EmployeeMapper, Employee>
        implements IEmployeeService {
}

2.分页的操作

1.在Employee实体类中加一个字段

@TableField(exist = false)
private Department department;

创建EmployeeQuery类

@Getter
@Setter
public class EmployeeQuery {
    private Integer currentPage=1;
    private Integer PageSize=3;
}

2.在IEmployeeService中添加抽象方法

IPage<Employee> query(EmployeeQuery qo);

3.在EmployeeServiceImpl类实现方法

@Override
    public IPage<Employee> query(EmployeeQuery qo) {
        IPage<Employee> page = new Page<>(qo.getCurrentPage(), qo.getPageSize());  //设置分页信息
        QueryWrapper<Employee> wrapper = Wrappers.<Employee>query();  //拼接条件
        return super.page(page,wrapper);
    }

4.测试类

@Test
    public void testPage(){
        EmployeeQuery qo = new EmployeeQuery();
        qo.setPageSize(3);
        qo.setCurrentPage(2);
        IPage<Employee> page = employeeService.query(qo);
        System.out.println("当前页:" + page.getCurrent());
        System.out.println("总页数:" + page.getPages());
        System.out.println("每页显示条数:" + page.getSize());
        System.out.println("总记录数:" + page.getTotal());
        System.out.println("当前页显示记录:" + page.getRecords());
    }

ppers.query(); //拼接条件
return super.page(page,wrapper);
}


4.测试类

```java
@Test
    public void testPage(){
        EmployeeQuery qo = new EmployeeQuery();
        qo.setPageSize(3);
        qo.setCurrentPage(2);
        IPage<Employee> page = employeeService.query(qo);
        System.out.println("当前页:" + page.getCurrent());
        System.out.println("总页数:" + page.getPages());
        System.out.println("每页显示条数:" + page.getSize());
        System.out.println("总记录数:" + page.getTotal());
        System.out.println("当前页显示记录:" + page.getRecords());
    }
举报

相关推荐

0 条评论