3-Mybatis高级

1.Mybatis 注解开发

2.Mybatis 注解实现多表操作

3.Mybatis 构建SQL语句

4.Mybatis 案例实现

1.Mybatis 注解开发

  • 常用注解
    @Select("查询的SQL语句"): 执行查询操作注解
    @Insert("新增的SQL语句"): 执行新增操作注解
    @Update("修改的SQL语句"): 执行修改操作注解
    @Delete("删除的SQL语句"): 执行删除操作注解
  • 配置映射关系
    <!--配置映射关系-->
    <mappers>
        <package name="接口所在包"/>
    </mappers>

注解实现查询操作
1.创建接口和查询方法
2.在核心配置文件中配置映射关系
3.编写测试类

  • 封装类
public class Student {
   private Integer id;
   private String name;
   private Integer age;
  • 接口类-注解方法编写sql语句
public interface StudentMapper {
    //查询全部的操作
    @Select("SELECT * FROM student")
    public abstract List<Student> selectAll();
}
  • 核心配置文件,只需添加配置文件,不需要加载映射配置文件标签
    <!--配置映射关系-->
    <mappers>
        <package name="com.itheima.mapper"/>
    </mappers>
  • 测试单元
public class Test01 {
    @Test
    public void selectAll() throws IOException {
        //1.加载核心配置文件
        InputStream is = Resources.getResourceAsStream("MyBatisConfig.xml");
        //2.获取SqlSession工厂类对象
        SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(is);
        //3.通过工厂类对象获取SqlSession对象
        SqlSession sqlSession = sqlSessionFactory.openSession(true);
        //4.获取StudentMapper接口的实现类对象
        StudentMapper mapper = sqlSession.getMapper(StudentMapper.class);
        //调用实现类中的方法,接收结果
        List<Student> list = mapper.selectAll();
        //处理结果
        for (Student student : list) {
            System.out.println(student);
        }
        sqlSession.close();
        is.close();
    }
}
  • 测试结果
    测试结果

注解实现新增操作
1.创建接口和新增方法
2.在核心配置文件中配置映射关系
3.编写测试类

  • 封装类不变
  • 接口类- 新增注解方法编写SQL语句
    //新增操作
    @Insert("insert into student values(#{id},#{name},#{age})")
    public abstract Integer insert(Student stu);
  • 核心配置文件不变
  • 测试单元
    @Test
    public void insert() throws IOException {
        //注意: 这里前面四个步骤和查询的测试单元一样,我这里省略了..........
        //调用实现类中的方法,接收结果
        Student stu1 = new Student(12, "金吒", 555);
        Integer insert = mapper.insert(stu1);
        //处理结果
        System.out.println(insert);
        sqlSession.close();
        is.close();
    }
  • 测试结果
    测试结果

注解实现修改操作

  • 封装类不变
  • 接口类- 修改注解方法编写SQL语句
    //修改操作
    @Update("update student set name=#{name},age=#{age} where id=#{id}")
    public abstract Integer update(Student stu);
  • 核心配置文件不变
  • 测试单元
 @Test
    public void upadte() throws IOException {
        //注意: 这里前面四个步骤和查询的测试单元一样,我这里省略了..........
        //调用实现类中的方法,接收结果
        Student stu1 = new Student(12, "哪吒", 18);
        Integer result= mapper.update(stu1);
        //处理结果
        System.out.println(result);
        sqlSession.close();
        is.close();
    }
  • 测试结果
    测试结果

注解实现删除操作

  • 封装类不变
  • 接口类- 删除注解方法编写SQL语句
    //删除操作
    @Delete("delete from student where id=#{id}")
    public abstract Integer delete(Integer id);
  • 核心配置文件不变
  • 测试单元
    @Test
    public void delete() throws IOException {
        //注意: 这里前面四个步骤和查询的测试单元一样,我这里省略了..........
        //调用实现类中的方法,接收结果
        Integer result = mapper.delete(12);
        //处理结果
        System.out.println(result);
        sqlSession.close();
        is.close();
    }
  • 测试结果
    测试结果

2.Mybatis 注解实现多表操作

一对一

  • @Results: 封装映射关系的父注解
    Result[] value(): 定义了Result数组
  • @Restult: 封装映射关系的子注解
    column属性: 查询出的表中字段名称
    property属性: 实体对象中的属性名称
    javaType属性: 被包含对象的数据类型
    one属性: 一对一查询固定属性
  • @One: 一对一查询的注解
    select属性: 指定调用某个接口中的方法
  • 封装对象
//get,set,toString等方法省略
public class Card {
    private Integer id;
    private String number;
    private  Person P;
}
//get,set,toString等方法省略
public class Person {
    private Integer id;
    private String name;
    private Integer age;
}
  • 核心配置文件
    <!--配置映射关系-->
    <mappers>
        <package name="com.itheima.one_to_one"/>
    </mappers>
  • 查询全部接口
//一对一注解开发方式
public interface CardMapper {
    //查询全部
    @Select("select * from card")
    @Results({
     @Result(column = "id",property = "id"),
     @Result(column = "number",property = "number"),
     @Result(
             property = "P",            //被包含对象的变量名
             javaType = Person.class,   //被包含对象的实际数据类型
             column = "pid",            //根据查询处的card表中的pid字段,来查询person表
             /*
             one,@One: 一对一查询的固定写法
             select属性: 指定调用哪个接口中的哪个方法
              */
             one=@One(select ="com.itheima.one_to_one.PersonMapper.selectById" )
        )
    })
    public abstract List<Card> selectAll();

}
  • 根据id查询接口
public interface PersonMapper {
    //根据id查询
    @Select("select * from person where id=#{id}")
    public abstract Person selectById(Integer id);

}
  • 测试单元
public class Test01 {
    @Test
    public void selectAll() throws IOException {
        //加载核心配置文件
        InputStream is = Resources.getResourceAsStream("MyBatisConfig.xml");
        //获取SqlSession工厂类对象
        SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(is);
        //通过工厂类对象获取SqlSession对象
        SqlSession sqlSession = sqlSessionFactory.openSession(true);
        //获取StudentMapper接口的实现类对象
        CardMapper mapper = sqlSession.getMapper(CardMapper.class);
        //调用实现类中的方法,接收结果
        List<Card> list = mapper.selectAll();
        //处理结果
        for (Card card : list) {
            System.out.println(card);
        }
        sqlSession.close();
        is.close();
    }
}
  • 测试结果
    测试结果

一对多

  • many属性: 一对多查询固定属性
  • @Many: 一对多查询的注解
    select属性: 指定调用某个接口中的方法
  • 封装类
//get,set,等方法省略
public class Person {
    private Integer id;
    private String name;
    private Integer age;
}
//get,set,等方法省略
public class Classes {
    private Integer id; //主键id
    private  String name;   //班级名称
    private List<Student> students; //班级中所有学生对象
  • 一对多的注解开发的接口
public interface ClassesMapper {
    //查询全部
    @Select("Select * from classes")
    @Results({
            @Result(column = "id",property = "id"),
            @Result(column = "name",property = "name"),
            @Result(
                    property = "students",  //被包含对象的变量名
                    javaType = List.class,  //被包含对象的数据类型
                    column = "id",   //根据查询处的classes表的id字段来查询student表
                    /*
                    一对多固定的写法
                     */
                    many=@Many(select = "com.itheima.one_to_many.StudentMapper.selectById")
            )
    })
    public abstract List<Classes> selectAll();
}
  • 以ClassMapper接口的查询结果的id,作为StudentMapper接口中查询方法的参数
public interface StudentMapper {
    //根据id查询
    @Select("Select * from student where cid=#{id}")
    public abstract List<Student> selectById(Integer id);
}
  • 测试单元
public class Test1 {
    @Test
    public void selectAll() throws IOException {
        //加载核心配置文件
        InputStream is = Resources.getResourceAsStream("MyBatisConfig.xml");
        //获取SqlSession工厂类对象
        SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(is);
        //通过工厂类对象获取SqlSession对象
        SqlSession sqlSession = sqlSessionFactory.openSession(true);
        //获取StudentMapper接口的实现类对象
        ClassesMapper mapper = sqlSession.getMapper(ClassesMapper.class);
        //调用实现类中的方法,接收结果
        List<Classes> list = mapper.selectAll();
        //处理结果
        for (Classes cls : list) {
            System.out.println(cls.getId()+","+cls.getName());
            List<Student> stu = cls.getStudents();
            for (Student student : stu) {
                System.out.println(student);
            }
        }
        sqlSession.close();
        is.close();
    }
}
  • 测试结果
    测试结果

多对多查询

  • 封装类
public class Course {
    private Integer id; //主键id
    private String name;   //课程名称
}
public class StudentAddCource {
    private Integer id; //主键id
    private String name;    //学生新明
    private Integer age;    //学生年龄
    private List<Course> courses;   //指当前学生所选择的课程集合
}
  • 接口
public interface CourseMapper {
    //根据id查询
    @Select("select c.id,c.name from stu_cr sc,course c where sc.cid=c.id and sc.sid=#{id}")
    public  abstract List<Course> selectById();
}
  • 查询所有学生接口,通过结果的id,调用CourseMapper中的根据id查询的方法
public interface StudentMapper {
    //查询全部
    @Select("SELECT DISTINCT s.id,s.name,s.age FROM student s, stu_cr sc WHERE sc.sid=s.id")
  //这条语句主要是筛选掉没有选课的学生
    @Results({
            @Result(column = "id",property = "id"),
            @Result(column = "name",property = "name"),
            @Result(column = "age",property = "age"),
            @Result(
                    property = "courses",
                    javaType = List.class,
                    column = "id",
                    many =@Many(select = "com.itheima.many_to_many.CourseMapper.selectById")
            )
    })
    public  abstract List<StudentAddCource> selectAll();
}
  • 测试单元
 @org.junit.Test
    public void selectAll() throws IOException {
        //加载核心配置文件
        InputStream is = Resources.getResourceAsStream("MyBatisConfig.xml");
        //获取SqlSession工厂类对象
        SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(is);
        //通过工厂类对象获取SqlSession对象
        SqlSession sqlSession = sqlSessionFactory.openSession(true);
        //获取StudentMapper接口的实现类对象
        StudentMapper mapper = sqlSession.getMapper(StudentMapper.class);
        //调用实现类中的方法,接收结果
        List<StudentAddCource> studentAddCources = mapper.selectAll();
        //处理结果
        for (StudentAddCource cls : studentAddCources) {
            System.out.println(cls.getId()+","+cls.getName()+","+cls.getAge());
            List<Course> courses = cls.getCourses();
            for (Course cours : courses) {
                System.out.println("\t"+cours);
            }
        }
        sqlSession.close();
        is.close();
    }
  • 测试结果
    测试结果

3.Mybatis 构建SQL语句

  • org.apache.ibatis.jdbc.SQL功能类,专门用于构建SQL语句


    常用方法
  • 定义功能类并提供获取查询的SQL语句的方法
  • @SelectProvider: 生成查询用的SQL语句注解
    type属性: 生成SQL语句功能类对象
    method属性: 指定调用方法
//定义方法,返回查询的SQL语句
    public String getSelectAll(){
        String sql = new SQL() {
            {
                SELECT("*");
                FROM("student");
            }
        }.toString();
        return sql;
    }
  • 定义功能类并提供获取新增的SQL语句的方法
  • @InsertProvider: 生成新增用的SQL语句注解
    type属性: 生成SQL语句功能类对象
    method属性: 指定调用方法
//定义方法,返回添加的SQL语句
    public String getInsert(Student stu){
        return new SQL() {
            {
                INSERT_INTO("student");
                INTO_VALUES("#{id},#{name},#{age}");
            }
        }.toString();
    }
  • 定义功能类并提供获取修改的SQL语句的方法
  • @UpdateProvider: 生成修改用的SQL语句注释
    type属性: 生成SQL语句功能类对象
    method属性: 指定调用方法
//定义方法,返回修改的SQL语句
    public String getUpdate(Student stu){
        return new SQL() {
            {
                UPDATE("student");
                SET("name=#{name},age=#{age}");
                WHERE("id=#{id}");
            }
        }.toString();
    }
  • 定义功能类并提供获取删除的SQL语句的方法
  • DeleteProvider: 生成删除用的SQL语句注解
    type属性: 生成SQL语句功能类对象
    method属性: 指定调用方法
//定义方法,返回修改的SQL语句
    public String getDelete(Integer id){
        return new SQL() {
            {
                DELETE_FROM("student");
                WHERE("id=#{id}");
            }
        }.toString();
    }

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容