Mybatis表现关联关系比hibernate简单,没有分那么细致one-to-many、many-to-one、one-to-one。而是只有两种association(一)、collection(多),表现很简洁。下面通过一个实例,来展示一下Mybatis对于常见的一对多和多对一关系复杂映射是怎样处理的。

以最简单的用户表订单表这个最简单的一对多做示例:
对应的JavaBean:
User:
public class User {
private int id;
private String name;
private Double age;
private List<User_orders> orders;
// get set 省
}
User_orders:
public class User_orders {
private int id;
private String name;
// get set 省
}
对应的数据库:
mysql> desc user; +-------+-------------+------+-----+---------+----------------+ | Field | Type | Null | Key | Default | Extra | +-------+-------------+------+-----+---------+----------------+ | id | int(11) | NO | PRI | NULL | auto_increment | | name | varchar(20) | NO | | NULL | | | age | double | YES | | NULL | | +-------+-------------+------+-----+---------+----------------+ 3 rows in set (0.00 sec) mysql> desc user_orders; +---------+-------------+------+-----+---------+----------------+ | Field | Type | Null | Key | Default | Extra | +---------+-------------+------+-----+---------+----------------+ | id | int(11) | NO | PRI | NULL | auto_increment | | name | varchar(20) | NO | | NULL | | | user_id | int(5) | YES | MUL | NULL | | +---------+-------------+------+-----+---------+----------------+ 3 rows in set (0.00 sec)
现在查询一个user的id查询出所有信息.如果不考虑关联查询,我们会先根据user的id在user表中查询出name,age然后设置给User类的时候,再根据该user的id在user_orders表中查询出所有订单并设置给User类。这样的话,在底层最起码调用两次查询语句,得到需要的信息,然后再组装User对象。
嵌套语句查询
mybatis提供了一种机制,叫做嵌套语句查询,可以大大简化上述的操作,加入配置及代码如下:
<resultMap type="domain.User" id="user">
<id column="id" property="id"/>
<result column="age" property="age"/>
<collection column="id" property="orders" ofType="domain.User_orders"
select="selectOrderByUser">
<id column="id" property="id"/>
<result column="name" property="name"/>
</collection>
</resultMap>
<select id="selectOrderByUser" parameterType="integer" resultType="domain.User_orders">
select id,name from user_orders where user_id = #{id}
</select>
<select id="findById" resultMap="user" parameterType="integer">
select * from user where id = #{id}
</select>
测试(可以成功查询到所有信息):
String config = "sqlMapConfig.xml";
InputStream inputStream = Resources.getResourceAsStream(config);
sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
SqlSession session = sqlSessionFactory.openSession();
// 执行在bean配置文件中定义的sql语句
User user = session.selectOne("UserMapper.findById", 1);
//一句即可获取到复杂的User对象。
System.out.println(user);
session.commit();
session.close();
嵌套语句查询的原理
在上面的代码中,Mybatis会执行以下流程:
1.先执行 findById 对应的语句从User表里获取到ResultSet结果集;
2.取出ResultSet下一条有效记录,然后根据resultMap定义的映射规格,通过这条记录的数据来构建对应的一个User 对象。
当要对User中的orders属性进行赋值的时候,发现有一个关联的查询,此时Mybatis会先执行这个select查询语句,得到返回的结果,将结果设置到user的orders属性上
这种关联的嵌套查询,有一个非常好的作用就是:可以重用select语句,通过简单的select语句之间的组合来构造复杂的对象。想如上的两个select完全可以独立使用。
嵌套查询的多对一
上面的关联查询查询其实是对于一对多的查询,即从user中查出user_order的信息。
现在从user_order中查user的信息.
在User_order表中增加字段user:
public class User_orders {
private int id;
private String name;
private User user;
//xxx
}
配置select:
<resultMap type="domain.User_orders" id="user_order">
<id column="id" property="id"/>
<result column="name" property="name"/>
<association property="user" column="user_id" javaType="domain.User" select="selectUserByOrderId">
<id column="id" property="id"/>
<result column="age" property="age"/>
</association>
</resultMap>
<select id="selectUserByOrderId" parameterType="INTEGER" resultType="domain.User">
select id,age from user where id = #{id}
</select>
<select id="findOne" resultMap="user_order" parameterType="integer">
select * from user_orders where id=#{id}
</select>
测试:
SqlSession session = sqlSessionFactory.openSession();
// 执行在bean配置文件中定义的sql语句
User_orders user_orders= session.selectOne("User_ordersMapper.findOne", 1);
System.out.println(user_orders);
//查询到了user_order对应的user的信息
session.commit();
session.close();
嵌套查询的N+1问题
尽管嵌套查询大量的简化了存在关联关系的查询,但它的弊端也比较明显:即所谓的N+1问题。关联的嵌套查询显示得到一个结果集,然后根据这个结果集的每一条记录进行关联查询。
现在假设嵌套查询就一个(即resultMap 内部就一个association标签),现查询的结果集返回条数为N,那么关联查询语句将会被执行N次,加上自身返回结果集查询1次,共需要访问数据库N+1次。如果N比较大的话,这样的数据库访问消耗是非常大的!所以使用这种嵌套语句查询的使用者一定要考虑慎重考虑,确保N值不会很大。
以上面一对多(根据user的id查询order)的例子为例,select 语句本身会返回user条数为1 的结果集,由于它存在有1条关联的语句查询,它需要共访问数据库 1*(1+1)=2次数据库。
嵌套结果查询
嵌套语句的查询会导致数据库访问次数不定,进而有可能影响到性能。Mybatis还支持一种嵌套结果的查询:即对于一对多,多对多,多对一的情况的查询,Mybatis通过联合查询,将结果从数据库内一次性查出来,然后根据其一对多,多对一,多对多的关系和ResultMap中的配置,进行结果的转换,构建需要的对象。
重新定义User的结果映射 resultMap
<resultMap type="domain.User" id="user_auto"> <id column="id" property="id"/> <result column="age" property="age"/> <collection column="id" property="orders" ofType="domain.User_orders"> <id column="order_id" property="id"/> <result column="name" property="name"/> </collection> </resultMap>
对应的sql语句如下:
<select id="findAuth" resultMap="user_auto"> select u.id,u.age,o.id as order_id ,o.name,o.user_id as user_id from user u left outer join user_orders o on o.user_id = u.id </select>
嵌套结果查询的执行步骤:
1.根据表的对应关系,进行join操作,获取到结果集;
对于关联的结果查询,如果是多对一的关系,则通过形如 <association property="user" column="user_id" javaType="domain.User" > 进行配置,Mybatis会通过column属性对应的user_id 值去从内存中取数据,并且封装成User_order对象;
如果是一对多的关系,就如User和User_order之间的关系,通过形如 <collection column="id" property="orders" ofType="domain.User_orders">进行配置,MyBatis通过 id去内存中取User_orders对象,封装成List;
对于关联结果的查询,只需要查询数据库一次,然后对结果的整合和组装全部放在了内存中。
以上是通过查询User表所有信息来演示了一对多和多对一的映射对象处理。希望对大家的学习有所帮助,也希望大家多多支持。
# mybatis嵌套查询
# mybatis
# 结果嵌套查询
# mybatis嵌套查询语句
# MyBatis一对多嵌套查询的完整实例
# Java之Mybatis多层嵌套查询方式
# mybatis之嵌套查询和嵌套结果有哪些区别
# Mybatis的collection三层嵌套查询方式(验证通过)
# mybatis中嵌套查询的使用解读
# mybatis 一对多嵌套查询的实现
# MyBatis嵌套查询collection报错:org.apache.ibatis.exceptio
# 最简单
# 配置文件
# 会先
# 数为
# 中取
# 形如
# 关联关系
# 放在
# 一句
# 将会
# 有可能
# 两种
# 两次
# 这条
# 是怎样
# 然后再
# 非常好
# 只需要
# 为例
# 要对
相关文章:
电商平台网站制作流程,电商网站如何制作?
如何选择最佳自助建站系统?快速指南解析优劣
实惠建站价格推荐:2025年高性价比自助建站套餐解析
完全自定义免费建站平台:主题模板在线生成一站式服务
如何零成本快速生成个人自助网站?
网站app免费制作软件,能免费看各大网站视频的手机app?
如何在阿里云服务器自主搭建网站?
网站制作公司,橙子建站是合法的吗?
海南网站制作公司有哪些,海口网是哪家的?
专业型网站制作公司有哪些,我设计专业的,谁给推荐几个设计师兼职类的网站?
公司网站建设制作费用,想建设一个属于自己的企业网站,该如何去做?
免费的流程图制作网站有哪些,2025年教师初级职称申报网上流程?
可靠的网站设计制作软件,做网站设计需要什么样的电脑配置?
外贸公司网站制作,外贸网站建设一般有哪些步骤?
非常酷的网站设计制作软件,酷培ai教育官方网站?
上海制作企业网站有哪些,上海有哪些网站可以让企业免费发布招聘信息?
高端建站三要素:定制模板、企业官网与响应式设计优化
官网网站制作腾讯审核要多久,联想路由器newifi官网
制作企业网站建设方案,怎样建设一个公司网站?
建站之星安装后界面空白如何解决?
如何通过VPS搭建网站快速盈利?
建站之星如何助力企业快速打造五合一网站?
关于BootStrap modal 在IOS9中不能弹出的解决方法(IOS 9 bootstrap modal ios 9 noticework)
如何通过PHP快速构建高效问答网站功能?
美食网站链接制作教程视频,哪个教做美食的网站比较专业点?
javascript中的try catch异常捕获机制用法分析
网站制作公司广州有几家,广州尚艺美发学校网站是多少?
家族网站制作贴纸教程视频,用豆子做粘帖画怎么制作?
山东云建站价格为何差异显著?
建站主机CVM配置优化、SEO策略与性能提升指南
建站之星如何配置系统实现高效建站?
如何确保FTP站点访问权限与数据传输安全?
昆明网站制作哪家好,昆明公租房申请网上登录入口?
北京营销型网站制作公司,可以用python做一个营销推广网站吗?
西安专业网站制作公司有哪些,陕西省建行官方网站?
网站制作公司排行榜,四大门户网站排名?
如何挑选最适合建站的高性能VPS主机?
建站之星云端配置指南:模板选择与SEO优化一键生成
制作旅游网站html,怎样注册旅游网站?
ppt制作免费网站有哪些,ppt模板免费下载网站?
如何在腾讯云服务器上快速搭建个人网站?
番禺网站制作公司哪家值得合作,番禺图书馆新馆开放了吗?
免费制作海报的网站,哪位做平面的朋友告诉我用什么软件做海报比较好?ps还是cd还是ai这几个软件我都会些我是做网页的?
宿州网站制作公司兴策,安徽省低保查询网站?
建站之星2.7模板快速切换与批量管理功能操作指南
建站之星安装需要哪些步骤及注意事项?
如何快速配置高效服务器建站软件?
阿里云高弹*务器配置方案|支持分布式架构与多节点部署
如何自定义建站之星网站的导航菜单样式?
西安大型网站制作公司,西安招聘网站最好的是哪个?
*请认真填写需求信息,我们会在24小时内与您取得联系。