8.Mapper动态代理
在前面例子中自定义 Dao 接口实现类时发现一个问题:Dao 的实现类其实并没有干什么 实质性的工作,
它仅仅就是通过 SqlSession 的相关 API 定位到映射文件 mapper 中相应 id 的 SQL 语句,
真正对 DB 进行操作的工作其实是由框架通过 mapper 中的 SQL 完成的。
所以,MyBatis 框架就抛开了 Dao 的实现类,直接定位到映射文件 mapper 中的相应 SQL语句,对 DB 进行操作。
这种对 Dao 的实现方式称为 Mapper 的动态代理方式。
Mapper 动态代理方式无需程序员实现 Dao 接口。接口是由 MyBatis 结合映射文件自动 生成的动态代理实现的。
1.映射文件的 namespace 属性值
一般情况下,一个Dao接口的实现类方法使用的是同一个SQL映射文件中的SQL映射id
所以,MyBatis框架要求,将映射文件中<mapper/>标签的namespace 属性设为 Dao接口的全类名,
则系统会根据方法所属Dao接口,自动到相应 namespace 的映射文件中查找相关的SQL映射
<!-- 使用 mapper 进行动态代理的前提之一:需要将 mapper 的命名空间 定义为 接口的完整类名 -->
<!-- 系统会根据方法所属DAO接口,自动到响应的 namespace 的映射文件中,查找相关的SQL映射 --> <mapper namespace="com.mybatis.dao.StudentDAO" >
2.修改日志输出控制文件
3.Dao接口方法名
MyBatis 框架要求,接口中的方法名,与映射文件中相应的SQL 标签的id值相同。
系统会自动根据方法名到相应的映射文件中查找同名的SQL映射d
@Test
public void testAdd() {
//使用 动态 mapper 的前提之二:DAO中的方法,需要跟 mapper.xml 文件的 的 statement的 id 相对应
studentDAO.insertStudent(s);
sqlSession.commit();
}
4.Dao对象的获取
使用时,只需调用SqlSession的getMapper方法,即可获取指定接口的实现类对象
该方法的参数为指定Dao接口类的class值
5.删除Dao实现类
由于通过调用 Dao 接口的方法,不仅可以从 SQL 映射文件中找到所要执行 SQL 语句,还可通过方法参数及返回值,
将 SQL 语句的动态参数传入,将查询结果返回。所以,Dao 的 实现工作,完全可以由 MyBatis 系统自动根据映射文件完成。
所以,Dao 的实现类就不再需 要了。Dao 实现对象是由 JDK 的 Proxy 动态代理自动生成的
6.修改测试类
Student s = null;
private SqlSession sqlSession;
private StudentDAO studentDAO;
@Before
public void before() {
s= new Student();
s.setAge(25);
s.setName("xzk5");
s.setScore(99); sqlSession = SqlSessionUtil.getSqlSession();
studentDAO = sqlSession.getMapper(StudentDAO.class);
} @After
public void after() {
if(sqlSession != null) {
sqlSession.close();
}
} //动态代理的过程
//1.首先通过 sqlSession.getMapper方法 ,创建代理对象,对studentDAO 进行代理
//2.通过接口名定位到映射文件 mapper
//3.通过接口的 方法名 定位到 映射文件 mapper中 对应的 SQL语句
//Mapper 动态代理方式 无需程序员 实现 DAO 接口
//接口是由MyBatis 结合 映射文件自动生成的动态代理实现的 ,(这个代理对象就是dao的实现) @Test
public void testAdd() {
//使用 动态 mapper 的前提之二:DAO中的方法,需要跟 mapper.xml 文件的 的 statement的 id 相对应
studentDAO.insertStudent(s);
sqlSession.commit();
}
多查询条件无法接收的问题
在实际工作中,表单中所给出的查询条件有时是无法将其封装为一个对象的,也就是说,查询方法只能携带多个参数,
而不能携带将这多个参数进行封装的一个对象。对于这个问题, 有两种解决方案
(1)将多个参数封装为一个Map
@Test
public void testSelectStudentsByMap() {
Map<String,Object> map = new HashMap<String,Object>();
map.put("conName", "xx");
map.put("conScore", "59");
map.put("conStu", s); //s 是一个 student对象
List<Student> students = studentDAO.selectStudentsByMap(map);
for(Student student:students) {
System.out.println(student);
} }
<select id="selectStudentsByMap" resultType="com.mybatis.model.Student">
select * from student where name like '%' #{conName} '%' and score > #{conScore} and age > #{conStu.age}
<!-- #{} 可以 1.放入 Map的key 2. 对象的属性-->
</select>
(2)多个参数逐个接收
对于mapper中的SQL语句,而已通过参数索引 #{index}的方式逐个接收每个参数
<select id="selectStudentsByMap" resultType="com.mybatis.model.Student">
<!-- select * from student where name like '%' #{conName} '%' and score > #{conScore} and age > #{conStu.age} -->
<!-- #{} 可以放入 1. Map的key 2. Map的value的对象的属性--> select * from student where name like '%' #{0} '%' and age > #{1}
<!-- #{} 可以放入 3.参数的索引-->
<!-- 另外:4.可以放入任意字符,构成占位符 5.放入对象的属性 --></select>
@Test
public void testSelectStudentsByConditions() {
List<Student> students = studentDAO.selectStudentsByConditions("xx",24);
for(Student student:students) {
System.out.println(student);
} }
8.Mapper动态代理的更多相关文章
- Mybatis之旅第二篇-Mapper动态代理方式
一.引言 通过上一篇mybatis的入门学习,我们已经会使用mybatis实现简单的增删改查,但是我们也发现了用原始Dao开发的一些问题: Dao方法体存在重复代码:通过SqlSessionFacto ...
- Mybaits之Mapper动态代理开发
Mybaits之Mapper动态代理开发 开发规范: Mapper接口开发方法只需要程序员与Mapper接口(相当于Dao接口),由Mybatis框架根据接口定义创建接口的动态代理对象,代理对象的方法 ...
- Mybatis框架三:DAO层开发、Mapper动态代理开发
这里是最基本的搭建:http://www.cnblogs.com/xuyiqing/p/8600888.html 接下来做到了简单的增删改查:http://www.cnblogs.com/xuyiqi ...
- 【Mybatis】-- Mapper动态代理开发注意事项
1.1. Mapper动态代理方式 1.1.1. 开发规范 Mapper接口开发方法只需要程序员编写Mapper接口(相当于Dao接口),由Mybatis框架根据接口定义创建接口的动态代理对象,代理对 ...
- MyBatis开发Dao层的两种方式(Mapper动态代理方式)
MyBatis开发原始Dao层请阅读我的上一篇博客:MyBatis开发Dao层的两种方式(原始Dao层开发) 接上一篇博客继续介绍MyBatis开发Dao层的第二种方式:Mapper动态代理方式 Ma ...
- mybatis由浅入深day01_5.3 Mapper动态代理方法
5.3 Mapper动态代理方法(程序员只需要写mapper接口(相当于dao接口)) 5.3.1 实现原理(mapper代理开发规范) 程序员还需要编写mapper.xml映射文件 程序员编写map ...
- Spring 整合Mybatis Mapper动态代理方法
先看项目目录结构 很清爽了 最重要的Spring的核心配置文件,看一下 <?xml version="1.0" encoding="UTF-8"?> ...
- Mybatis之Mapper动态代理
一.什么是Mapper的动态代理 采用Mapper动态代理方法只需要编写相应的Mapper接口(相当于Dao接口),那么Mybatis框架根据接口定义创建接口的动态代理对象,代理对象的方法体同Dao接 ...
- MyBatis使用Mapper动态代理开发Dao层
开发规范 Mapper接口开发方法只需要程序员编写Mapper接口(相当于Dao接口),由Mybatis框架根据接口定义创建接口的动态代理对象,代理对象的方法体同原始Dao接口实现类方法. Mappe ...
- MyBatis开发Dao的原始Dao开发和Mapper动态代理开发
目录 咳咳...初学者看文字(Mapper接口开发四个规范)属实有点费劲,博主我就废了点劲做了如下图,方便理解: 原始Dao开发方式 1. 编写映射文件 3.编写Dao实现类 4.编写Dao测试 Ma ...
随机推荐
- 用VLC读取摄像头产生RTSP流,DSS侦听并转发(二)
用VLC读取摄像头产生RTSP流,DSS侦听并转发(二) 之前介绍过<用VLC读取摄像头产生RTSP流,DSS主动取流转发(一)>本文介绍另一种方法. 摄像机地址是192.1.101.51 ...
- ios 自动去重
//resultArrM 数据原//_indexArray 过滤后的数据//MYSelectAreaModel 模型 /* 重定义索引 */ - (void)sy_indexArray{ /* 索 ...
- python中星号变量的几种特殊用法
python中星号变量的几种特殊用法 不知道大家知不知道在Python中,星号除了用于乘法数值运算和幂运算外,还有一种特殊的用法"在变量前添加单个星号或两个星号",实现多参数的传入 ...
- Cognos Framework操作记录:开发复杂报表
设计一张数据库的表:TEST_001_ADDRESS 表结构: | 编号 | 姓名 | 省 | 市 | 县 | 公司 | 部门 | 职位 | | ---- | --- | -- | - | - | - ...
- 分布式开发之:id生成器
一般分布式系统开发中不建议使用数据库自带的自增ID做id. 理由: 1.不方便分库分表.(TIDB时代待商榷) 2.不利于多机房多活部署. 那么如果不使用数据库的id.那怎么生成id呢. 1. Twi ...
- Java基础——值传递
值传递? 参数传递的值的拷贝,改变拷贝不影响原参数. 引用传递? 参数传递的是参数本身,改变形参,实参也改变. Java中是什么传递? Java中只有值传递 实际情况为什么不对呢? 1. 基本数据类型 ...
- [BAT] cmd 管理员权限 右键菜单 运行
将下面保存为 *.reg Windows Registry Editor Version 5.00 [HKEY_CLASSES_ROOT\Directory\Background\shell\runa ...
- [国外] 解决Windows10下google搜索连接不上,但其它网页都能成功登入的问题
我的电脑突然在学校不能登陆google了,在家可以连接但是非常慢. 在网上搜索发现是由于DNS出了问题,在此记录一下. 可以通过控制面板——网络和Internet——网络和共享中心——点击现在使用的网 ...
- pytorch API中sgd.py的学习记录
参考:PyTorch与caffe中SGD算法实现的一点小区别 其中公式(3)(4)的符号有问题 变量对应表 程序 参考文章 buf v momentum μ d_p Δf(θ) lr ξ p θ
- [Cometoj#3 B]棋盘_状压dp
棋盘 题目链接:https://cometoj.com/contest/38/problem/B?problem_id=1535 数据范围:略. 题解: 因为行数特别小,所以$dp$的时候可以状压起来 ...