MyBatis入门学习(五)

使用注解开发

1、面向接口编程

很多时候,我们会选择面向接口编程

  • 根本原因:解耦,可拓展,提高复用,分层开发中,上层不用管具体的实现,大家都遵守共同的标准,使得开发变得容易,规范性更好
  • 在一个面向对象的系统中,系统的各种功能是由许许多多的不同对象协作完成的,在这种情况下,各个对象内部是如何实现自己的,对系统设计人员来讲就不那么重要了;
  • 而各个对象之间的协作关系则成为系统设计的关键,小到不同类之间的通信,大到各模块之间的交互,在系统设计之初都是要最重考虑的,这个也是系统设计的主要工作内容。面向接口编程就是指按照这种思想来编程。

关于接口的理解

  • 接口从更深层次的理解,应该是定义(规范、约束)与实现的分离。
  • 接口的本身反映了系统设计人员对系统的抽象理解。
  • 接口有两类:
    • 第一类是对一个个体的抽象,它可对应为一个抽象体(abstract class);
    • 第二类是对一个个体某一方面的抽象,即形成一个抽象面(interface);
  • 一个个体有可能有多个抽象面,抽象体与抽象面是有区别的。

三个面向区别

  • 面向对象是指,我们考虑问题是,一对象为单位,考虑它的属性及方法;
  • 面向过程是指,我们考虑问题时,一一个具体的流程(事务过程)为单位,考虑它的实现;
  • 接口设计与非接口设计是针对复用技术而言,与面向对象(过程)不是一个问题,更对的体现就是对系统整体的架构;

2、使用注解开发

1、注解在接口上实现

    @Select("select * from user")
    List<User> getUserList();

2、需要在核心配置文件中绑定接口

    <!--绑定接口-->
    <mappers>
        <mapper class="com.star.mapper.UserMapper"/>
    </mappers>

3、测试

    @Test
    public void getUserList(){
        SqlSession session = MybatisUtils.getSession();
        UserMapper mapper = session.getMapper(UserMapper.class);
        List<User> userList = mapper.getUserList();
        for (User user : userList) {
            System.out.println(user);
        }
    }

测试结果:

本质:反射机制实现

底层:动态代理

3、CRUD

我们可以在工具类创建的时候实现自动提交事务


    public static SqlSession getSession(){
        return getSession(false);
    }

    public static SqlSession getSession(boolean flag){
        return sqlSessionFactory.openSession(flag);
    }

编写接口,增加注解

package com.star.mapper;

import com.star.pojo.User;
import org.apache.ibatis.annotations.*;

import java.util.List;

public interface UserMapper {

    @Select("select * from user")
    List<User> getUserList();

    @Select("select * from user where id = #{id}")
    User selectById(@Param("id") int id);

    @Select("select * from user where name=#{username} and pwd=#{pwd}")
    User selectByUsernamePwd(@Param("username") String username, @Param("pwd") String pwd);

    @Insert("insert into user(id,`name`,pwd) value (#{id},#{name},#{pwd})")
    int addUser(User user);

    @Update("update user set name=#{name},pwd=#{pwd} where id=#{id}")
    int updateUser(User user);

    @Delete("delete from user where id = #{id}")
    int deleteUserById(@Param("id") int id);
}

测试类还和之前一样!

注意:我们必须要将接口注册绑定到我们的核心配置文件中!

关于@Param() 注解

  • 基本类型的参数或者String类型,需要加上
  • 引用类型不需要加
  • 如果只有一个基本类型的话,可以忽略,但是建议大家都加上!
  • 我们在SQL中引用的就是我们这里的 @Param() 中设定的属性名!

多对一处理

多对一:多个学生对应一个老师!

对于学生这边而言:多个学生关联一个老师【多对一】

首先创建一个数据库

CREATE TABLE `teacher` (
  `id` INT(10) NOT NULL,
  `name` VARCHAR(30) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=INNODB DEFAULT CHARSET=utf8

INSERT INTO teacher(`id`, `name`) VALUES (1, '秦老师'); 

CREATE TABLE `student` (
  `id` INT(10) NOT NULL,
  `name` VARCHAR(30) DEFAULT NULL,
  `tid` INT(10) DEFAULT NULL,
  PRIMARY KEY (`id`),
  KEY `fktid` (`tid`),
  CONSTRAINT `fktid` FOREIGN KEY (`tid`) REFERENCES `teacher` (`id`)
) ENGINE=INNODB DEFAULT CHARSET=utf8

INSERT INTO `student` (`id`, `name`, `tid`) VALUES ('1', '小明', '1');
INSERT INTO `student` (`id`, `name`, `tid`) VALUES ('2', '小红', '1');
INSERT INTO `student` (`id`, `name`, `tid`) VALUES ('3', '小张', '1');
INSERT INTO `student` (`id`, `name`, `tid`) VALUES ('4', '小李', '1');
INSERT INTO `student` (`id`, `name`, `tid`) VALUES ('5', '小王', '1');

测试环境搭建

1、导入所需的依赖

2、新建实体类Teacher、Student

package com.star.pojo;

import lombok.Data;

@Data
public class Teacher {
    private int id;
    private String name;

}
package com.star.pojo;

import lombok.Data;

@Data
public class Student {
    private int id;
    private String name;
    //学生关联老师
    private Teacher teacher;

}

3、建立Mapper接口

package com.star.mapper;

import com.star.pojo.Student;

import java.util.List;

public interface StudentMapper {

    //获取所有的学生及学生对应的老师的信息
    List<Student> getStudents();
}

4、建立Mapper.xml

有两种方式

方式一:按照结果嵌套处理

    <!--
    获取所有的学生及学生对应老师的信息
        1、获取所有的学生信息
        2、获取所有的学生的tid,然后再去老师表中查询这个tid对应的老师
    -->

    <resultMap id="StudentMap" type="Student">
        <id property="id" column="sid"/>
        <result property="name" column="sname"/>
        <!--这里存在一个问题,tid是一个字段,而teacher是一个对象-->
        <!--使用关联标签 association   javaType:对象的类型-->
        <association property="teacher" javaType="Teacher">
            <id property="id" column="tid"/>
            <result property="name" column="tname"/>
        </association>
    </resultMap>

    <!--
    这个其实对应MySql中的联表查询;
    查询的sql结果,可能来自多个对象,将多个对象关联;
    需要考虑查询出来的字段到底是对象的哪个属性!
    -->
    <select id="getStudents" resultMap="StudentMap">
        select s.id sid,s.name sname,t.name tname,t.id tid
        from student s,teacher t
        where s.tid = t.id;
    </select>

方式二:按照查询嵌套处理

    <select id="getStudents" resultMap="StudentTeacher">
        select * from student
    </select>

    <!--连接子查询-->
    <resultMap id="StudentTeacher" type="Student">
        <result property="id" column="id"/>
        <result property="name" column="name"/>
        <association property="teacher" column="tid" javaType="Teacher" select="getTeacher"/>
    </resultMap>

    <!--自定义的子查询-->
    <select id="getTeacher" resultType="Teacher">
        select * from teacher where id = #{id}
    </select>

5、在核心配置文件中绑定注册我们的Mapper接口或者文件!

    <mappers>
        <package name="com.star.mapper"/>
    </mappers>

6、测试

package com.star.mapper;

import com.star.pojo.Student;
import com.star.utils.MybatisUtils;
import org.apache.ibatis.session.SqlSession;
import org.junit.Test;

import java.util.List;

public class StudentMapperTest {

    @Test
    public void getStudentsTest() {
        SqlSession session = MybatisUtils.getSession();
        StudentMapper mapper = session.getMapper(StudentMapper.class);
        List<Student> students = mapper.getStudents();
        for (Student student : students) {
            System.out.println(student);
        }
    }
}

测试结果:

一对多处理

一对多:一个老师对应多个学生(集合)

对于老师而言,一个老师有很多学生【一对多】

1、实体类

package com.star.pojo;

import lombok.Data;

import java.util.List;

@Data
public class Teacher {
    private int id;
    private String name;
    //老师对应的学生房子一个集合里
    private List<Student> students;

}
package com.star.pojo;

import lombok.Data;

@Data
public class Student {
    private int id;
    private String name;
    private int tid;

}

2、编写对应Mapper接口

package com.star.mapper;

import com.star.pojo.Teacher;

public interface TeacherMapper {

    //通过老师的id查询该老师对应的所有学生的信息
    public Teacher getTeacher(int id);
}

3、编写Mapeer.xml文件

有两种方式

方式一:按照结果嵌套处理

    <resultMap id="TeacherMap" type="Teacher">
        <id property="id" column="tid"/>
        <result property="name" column="tname"/>
        <!--
        这里的字段students是一个集合(包含的关系) 使用collection标签
        ofType:集合中的元素对应的Java类型
        -->
        <collection property="students" ofType="Student">
            <id property="id" column="sid"/>
            <result property="name" column="sname"/>
            <result property="tid" column="stid"/>
        </collection>
    </resultMap>

    <select id="getTeacher" resultMap="TeacherMap">
        select t.id tid,t.name tname,s.id sid,s.name sname,s.tid stid
        from student s,teacher t
        where s.tid=t.id and tid=#{id};
    </select>

方式二:按照查询嵌套处理

    <select id="getTeacher" resultMap="TeacherMap">
        select * from teacher where id = #{tid}
    </select>

    <!--连接子查询-->
    <resultMap id="TeacherMap" type="Teacher">
        <collection property="students" javaType="ArrayList" ofType="Student" select="getStudentByTeacherId" column="id"/>
    </resultMap>

    <!--自定义的一个子查询-->
    <select id="getStudentByTeacherId" resultType="Student">
        select * from student where tid = #{tid}
    </select>

4、核心配置文件绑定Mapper.xml

    <mappers>
        <package name="com.star.mapper"/>
    </mappers>

5、测试

package com.star.mapper;

import com.star.pojo.Teacher;
import com.star.utils.MybatisUtils;
import org.junit.Test;

public class TeacherMapperTest {

    @Test
    public void getTeacherTest(){
        TeacherMapper mapper = MybatisUtils.getSession().getMapper(TeacherMapper.class);
        Teacher teacher = mapper.getTeacher(1);
        System.out.println(teacher);
    }
}

测试结果:

结论:

  • 关联 association【多对一】
  • 集合 collection【一对多】
  • javaType & ofType
    • javaType 用来指定实体类中属性的类型
    • ofType 用来指定映射到List或者集合中的pojo类型,泛型中的约束类型

注意:

  • 保证SQL的可读性
  • 注意一对多和多对一中,属性名和字段的问题
  • 建议使用日志,LOG4J

原文地址:https://www.cnblogs.com/lmx-181028/p/12358904.html

时间: 2024-10-15 12:31:13

MyBatis入门学习(五)的相关文章

Mybatis入门学习篇(三)之模糊查询的两种写法

在上一讲(Mybatis入门学习篇(二)之基于注解的增删改查)中,需要用到模糊查询,比如我想查找所有含有'zjh'的Student的信息.如果是在数据库中,那么我们可以方便的使用通配符%:select * from student where name like '%zjh%' .但是如果在mybatis中直接像sql中那样,就会报错.遂百度,无果,偶然在一篇帖子中看到,遂记录下来,以备后用. 方法一: 在要查询的字符串合适位置加上%,如这里的'zjh'就应该为'%zjh%'.这是一个很管用的方

Objective C 快速入门学习五

<一>继承和多态 @class Complex 声明类(同C++) 子类函数成员 super 访问父类 同C++类似 1.通过继承 在子类中添加新方法 2.通过继承 在子类中添加新成员 3.通过继承 实现多态(实现比较简单,通过Id通用类型作为父类) 4.重载 5.抽象类abstract作用:创建子类更容易:提供了处理所有派生子类的公共接口:抽象方法制定了标准协议,规范子类必须实现. 6.通用类型id,编译时不会做类型检查,在运行时才会动态绑定具体类型,指出错误. 静态类型在编译阶段就会指出错

MyBatis入门学习(二)

在MyBatis入门学习(一)中我们完成了对MyBatis简要的介绍以及简单的入门小项目测试,主要完成对一个用户信息的查询.这一节我们主要来简要的介绍MyBatis框架的增删改查操作,加深对该框架的了解.环境测试是上一节中的测试环境.在此感谢博主提供的资料. 一.通过MyBatis持久层框架完成对数据库表的增删改查操作——基于XML的实现 1.定义sql文件的映射文件,userMapper文件内容如下: <?xml version="1.0" encoding="UTF

MyBatis入门(五)---延时加载、缓存

一.创建数据库 1.1.建立数据库 /* SQLyog Enterprise v12.09 (64 bit) MySQL - 5.7.9-log : Database - mybatis ********************************************************************* */ /*!40101 SET NAMES utf8 */; /*!40101 SET SQL_MODE=''*/; /*!40014 SET @[email protec

mybatis入门学习之(1)+简单例子测试

Mybatis 入门之(1)环境配置+简单例子测试 什么是MyBatis? 它是支持普通SQL查询,存储过程和高级映射的优秀持久层框架.MyBatis 消除了几乎所有的JDBC代码和参数的手工设置以及结果集的检索.MyBatis 使用简单的XML或注解用于配置和原始映射,将接口和POJOs(Plan Old Java Objects,普通的 Java对象)映射成数据库中的记录. 其实简单一点说mybatis就是一直访问数据库的技术,它是现在比较流行的一个持久层框架,如果你对JDBC熟悉那就更容易

mybatis入门学习

一.mybaits简介 java程序中常用的数据库框架. 二.mybait入门学习 1.环境 创建一个java或者javaweb项目:我是创建了一个structs项目,helloworld 2.添加相应的jar包

C语言程序设计入门学习五步曲(转发)

笔者在从事教学的过程中,听到同学抱怨最多的一句话是:老师,上课我也能听懂,书上的例题也能看明白,可是到自己动手做编程时,却不知道如何下手.发生这种现象的原因有三个: 一.所谓的看懂听明白,只是很肤浅的语法知识,而我们编写的程序或软件是要根据要解决问题的实际需要控制程序的流程,如果你没有深刻地理解C语言的语句的执行过程(或流程),你怎么会编写程序解决这些实际问题呢? 二.用C语言编程解决实际问题,所需要的不仅仅是C语言的编程知识,还需要相关的专业知识.例如,如果你不知道长方形的面积公式,即使C语言

Mybatis入门学习(一)

1.MyBatis简介 MyBatis 本是apache的一个开源项目iBatis, 2010年这个项目由apache software foundation 迁移到了google code,并且改名为MyBatis .2013年11月迁移到Github. 需要在Github上下载!地址:https://github.com/mybatis/mybatis-3 当前的最新版本是MyBatis 3.5.4! 我们可以在其官网上查看文档进行学习!地址:https://mybatis.org/myba

MyBatis入门学习教程-调用存储过程

一.提出需求 查询得到男性或女性的数量, 如果传入的是0就女性否则是男性 二.准备数据库表和存储过程 1 create table p_user( 2 id int primary key auto_increment, 3 name varchar(10), 4 sex char(2) 5 ); 6 7 insert into p_user(name,sex) values('A',"男"); 8 insert into p_user(name,sex) values('B',&qu