Mybatis中连接查询和嵌套查询实例代码

首先在mysql中确立表:

#表一:地址国家表
CREATE TABLE address(aid INT AUTO_INCREMENT PRIMARY KEY,aname VARCHAR(20));

INSERT INTO address VALUES(NULL,"魏国");
INSERT INTO address VALUES(NULL,"蜀国");
INSERT INTO address VALUES(NULL,"吴国");

#表二:出场人物表
CREATE TABLE person(
	pid INT AUTO_INCREMENT PRIMARY KEY,
	pname VARCHAR(20),
	paid INT,
	CONSTRAINT pafk FOREIGN KEY person(paid) REFERENCES address(aid) ON UPDATE CASCADE ON DELETE CASCADE
	);

INSERT INTO person VALUES(1,"曹操",1);
INSERT INTO person VALUES(2,"荀彧",1);
INSERT INTO person VALUES(3,"张辽",1);

INSERT INTO person VALUES(4,"刘备",2);
INSERT INTO person VALUES(5,"关羽",2);
INSERT INTO person VALUES(6,"张飞",2);
INSERT INTO person VALUES(7,"诸葛亮",2);

INSERT INTO person VALUES(8,"孙权",3);
INSERT INTO person VALUES(9,"周瑜",3);
INSERT INTO person VALUES(10,"陆逊",3);

INSERT INTO person VALUES(11,"公孙瓒",NULL);

#表三:交通工具表
CREATE TABLE tool(tid INT AUTO_INCREMENT PRIMARY KEY,tname VARCHAR(20));
INSERT INTO tool VALUES(1,"马");
INSERT INTO tool VALUES(2,"船");

#表四:地址国家——交通工具 多对多关系表
CREATE TABLE aandt(
	a_aid INT,
	a_tid INT,
	PRIMARY KEY(a_aid,a_tid),#联合主键,是指多个字段组成一个组合,该组合在数据表中唯一
	CONSTRAINT FOREIGN KEY aandt(a_aid) REFERENCES address(aid) ON UPDATE CASCADE ON DELETE CASCADE,
	CONSTRAINT FOREIGN KEY aandt(a_tid) REFERENCES tool(tid) ON UPDATE CASCADE ON DELETE CASCADE
);

INSERT INTO aandt VALUES(1,1);
INSERT INTO aandt VALUES(2,1);
INSERT INTO aandt VALUES(2,2);
INSERT INTO aandt VALUES(3,2);

查询a表的所有信息,如果a表的信息有对应的b表的信息,则查询b表的信息,如果没有,则不查询。

多对一,如:查询所有人物信息,如果人物有对应国家,则查询国家信息,如果没有,则不查询。多个人物属于一个国家。

一对多,如:查询所有国家信息,如果国家有对应人物,则查询人物信息,如果没有,则不查询。一个国家拥有多个城市。

多对多,如:查询所有国家信息,如果国家拥有对应的交通工具,则查询交通工具信息,没有则不查询。与此同时,多种交通工具存在于于多个国家。

一、连接查询:

连接查询使用时,使用偏向于a表所在方向的外连接,可获得a表所有信息,和对应的b表信息。该方式为饿汉式,内存占用较大,但对数据库访问次数较少而导致消耗时间少。

1、多对一:

<!--多对一的数据库-->
 <mapper namespace="com.fh.dao.PersonDao">
 	<!--该映射的id为map1,该映射的内容为:将查询到的字段的结果值,按照本映射的对应关系,分别封装在Person实体类下的各个属性上,整体构成Person-->
 <resultMap id="map1" type="com.fh.domain.Person">
  <id column="pid" property="pid"/><!--id:主键映射; column:数据库表字段; property:类中对应属性名-->
  <result column="pname" property="pname"/><!--result:非主键映射-->
  <result column="paid" property="paid"/>
  <association property="address" javaType="com.fh.domain.Address"><!--association:在查询到多,然后对应出一时,用于关联对应出一的一方; property:查询类中包含的子对象属性; javaType:子对象属性封装对应的类-->
   <id column="aid" property="aid"/>
   <result column="aname" property="aname"/>
  </association>
 </resultMap>
 <select id="findAllPerson" resultMap="map1">/*resultMap:自己编写的结果集,本查询的返回值正是该结果集所对应的映射组构成的Person*/
--   查询所有人及其对应的地址,以人为主,则对人侧外连接
  SELECT * FROM person p LEFT OUTER JOIN address a ON p.paid = a.aid
 </select>
</mapper>

2、一对多:

<!--一对多的数据库-->
 <mapper namespace="com.fh.dao.AddressDao">
 <resultMap id="map2" type="com.fh.domain.Address">
  <id column="aid" property="aid"/>
  <result column="aname" property="aname"/>
  <collection property="personList" ofType="com.fh.domain.Person"><!--collection:查询到一,接着关联对应出多时,指向多的一方; foType:集合中每个元素所属的映射类-->
   <id column="pid" property="pid"/>
   <result column="pname" property="pname"/>
   <result column="paid" property="paid"/>
  </collection>
 </resultMap>
 <select id="findAllAddress" resultMap="map2">
  SELECT * FROM address a LEFT OUTER JOIN person p ON a.aid = p.paid;
 </select>
</mapper>

3、多对多:

<mapper namespace="com.fh.dao.ToolDao">
 <resultMap id="map3" type="com.fh.domain.Tool">
  <id column="tid" property="tid"/>
  <result column="tname" property="tname"/>
  <collection property="addressList" ofType="com.fh.domain.Address">
   <id column="aid" property="aid"/>
   <result column="aname" property="aname"/>
  </collection>
 </resultMap>
 <select id="findAllTool" resultMap="map3">
  SELECT t.*,a.* FROM tool AS t LEFT OUTER JOIN aandt AS a_t ON t.`tid` = a_t.`a_tid` LEFT OUTER JOIN address AS a ON a_t.`a_aid` = a.`aid`;
 </select>

二、嵌套查询:

嵌套查询使用时,先查询a表的信息,然后依赖a和b表的外键约束,利用in(),再次查询b表对应到a表上的信息。该方式可以改为饿汉式,内存使用较小,但需要多次访问数据库而导致消耗时间多。

1、多对一:

PersonDao接口内写入:

//查询所有人,以及其对应的地址
 List<Person> findPersonFromAddress();

对应映射配置中:

<!--多对一的数据库-->
 <mapper namespace="com.fh.dao.PersonDao">
 <!--person映射的基本属性对应下面person的结果集,本结果集内部再继续进行处理-->
 <resultMap id="map1" type="com.fh.domain.Person">
  <id column="pid" property="pid"/>
  <result column="pname" property="pname"/>
  <result column="paid" property="paid"/>
  <!--
  对应到person内的子属性对象,column内为person对应到address的外键,由此外键,传入select内的方法进行二次嵌套查询,交由AddressDao在其中的指定方法进行处理
  即要求查询目标为:Address.aid = Person.paid
  -->
  <association property="address" column="paid" select="com.fh.dao.AddressDao.findAddressById"/>
 </resultMap>
 <!--第一次查询为只查询主要对象,自定义结果集-->
 <select id="findPersonFromAddress" resultMap="map1">
  select * from person
 </select>
</mapper>

继续编写指向AddressDao接口中的findAddressById:

//按照id查询Address
List<Address> findAddressById(Integer id);

回到AddressDao配置文件:

 <mapper namespace="com.fh.dao.AddressDao">
 <!--根据id对address查询-->
 <select id="findAddressById" resultType="com.fh.domain.Address">
  SELECT * FROM address WHERE aid = #{aid}
 </select>

2、一对多:

AddressDao接口内写入:

List<Address> findAddressWithPerson();

其对应映射配置中:

<resultMap id="map2" type="com.fh.domain.Address">
  <id column="aid" property="aid"/>
  <result column="aname" property="aname"/>
  <collection property="personList" column="aid" select="com.fh.dao.PersonDao.findPersonById"/>
 </resultMap>
 <select id="findAddressWithPerson" resultMap="map2">
  select * from address
 </select>

针对指出的PersonDao接口的findPersonById:

 List<Person> findPersonById(Integer id);

其对应的映射配置中:

<select id="findPersonById" resultType="com.fh.domain.Person">
  select * from person where pid = #{pid}
 </select>

对于嵌套查询的延迟加载问题,需添加配置:

方法一:

association或collection中多加一条属性:fetchType=“lazy”

方法二:

<settings>
  <setting name="lazyLoadingEnable" value="true"/>
  <setting name="lazyLoadTriggerMethods" value="true"/><!--本条设置表示将包括原本不会延迟加载的equals/clone/hashCode/toString在内所有方法进行延迟加载-->
 </settings>

到此这篇关于Mybatis中连接查询和嵌套查询的文章就介绍到这了,更多相关Mybatis连接查询和嵌套查询内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • MyBatis的嵌套查询解析

    Mybatis表现关联关系比hibernate简单,没有分那么细致one-to-many.many-to-one.one-to-one.而是只有两种association(一).collection(多),表现很简洁.下面通过一个实例,来展示一下Mybatis对于常见的一对多和多对一关系复杂映射是怎样处理的. 以最简单的用户表订单表这个最简单的一对多做示例: 对应的JavaBean: User: public class User { private int id; private String

  • Java的MyBatis框架中实现多表连接查询和查询结果分页

    实现多表联合查询 还是在david.mybatis.model包下面新建一个Website类,用来持久化数据之用,重写下相应toString()方法,方便测试程序之用. package david.mybatis.model; import java.text.SimpleDateFormat; import java.util.Date; public class Website { private int id; private String name; private int visito

  • mybatis同一张表多次连接查询相同列赋值问题小结

    最近遇到的一些问题总结: 1. MySQL数据库同一张表做四次左连接查询数据冗余. a. mysql数据库连接查询 b. mysql表数据去重 2. mybatis查询相同列赋值重复问题. a. 使用mybatis强大的resultMap b. mysql数据查询别名 数据库表为 Create Table CREATE TABLE `STUDENT_SCORE` ( `score_id` int(11) unsigned NOT NULL AUTO_INCREMENT,//主键 `test_id

  • Mybatis中连接查询和嵌套查询实例代码

    首先在mysql中确立表: #表一:地址国家表 CREATE TABLE address(aid INT AUTO_INCREMENT PRIMARY KEY,aname VARCHAR(20)); INSERT INTO address VALUES(NULL,"魏国"); INSERT INTO address VALUES(NULL,"蜀国"); INSERT INTO address VALUES(NULL,"吴国"); #表二:出场人物

  • Mybatis中的resultType和resultMap查询操作实例详解

    resultType和resultMap只能有一个成立,resultType是直接表示返回类型的,而resultMap则是对外部ResultMap的引用,resultMap解决复杂查询是的映射问题.比如:列名和对象属性名不一致时可以使用resultMap来配置:还有查询的对象中包含其他的对象等. MyBatisConfig.xml <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE configura

  • MYSQL子查询和嵌套查询优化实例解析

    查询游戏历史成绩最高分前100 Sql代码 SELECT ps.* FROM cdb_playsgame ps WHERE ps.credits=(select MAX(credits) FROM cdb_playsgame ps1 where ps.uid=ps1.uid AND ps.gametag=ps1.gametag) AND ps.gametag='yeti3' GROUP BY ps.uid order by ps.credits desc LIMIT 100; Sql代码 SEL

  • Mybatis 中 Oracle 的拼接模糊查询及用法详解

    一.结论 这里先给大家看一下结论 Oracle 中,拼接模糊查询的正确写法 SELECT A.USER_ID, A.USER_NAME FROM USER A AND A.USER_NAME like concat(concat('%','w'),'%') 或者 AND A.USER_NAME like '%' || 'w' || '%' Mybatis 中,拼接模糊查询的正确写法 <select id="selectByName" resultMap="BaseRes

  • 详解MySQL子查询(嵌套查询)、联结表、组合查询

    一.子查询 MySQL 4.1版本及以上支持子查询 子查询:嵌套在其他查询中的查询. 子查询的作用: 1.进行过滤: 实例1:检索订购物品TNT2的所有客户的ID = + 一般,在WHERE子句中对于能嵌套的子查询的数目没有限制,不过在实际使用时由于性能的限制,不能嵌套太多的子查询. 注意:列必须匹配 --在WHERE子句中使用子查询(如这里所示),应该保证SELECT语句具有与WHERE子句中相同数目的列.通常,子查询将返回单个列并且与单个列匹配,但如果需要也可以使用多个列. 示例2:返回订购

  • 使用Bootstrap框架制作查询页面的界面实例代码

    以Bootstrap框架来进行设计和开发,是目前国际上比较流行的一个趋势.很多软件公司在优化新产品时,因为其在js和控件上的综合优势,会选用这个开发框架. Bootstrap框架是一个前端UI设计的框架,它提供了统一的UI界面,简化了设计界面UI的过程(缺点是定制了界面,调整的余地不是太大).尤其是现在的响应时布局(我的理解是页面根据不同的分辨率,采用不同的页面元素的布局),在Bootstrap中很好的支持了,只要简单设置了属性,就能自动实现响应时布局,大大简化了程序员的界面的过程. 因此,本人

  • MongoDB快速入门笔记(四)之MongoDB查询文档操作实例代码

    MongoDB简介 MongoDB 是一个基于分布式文件存储的数据库.由 C++ 语言编写.旨在为 WEB 应用提供可扩展的高性能数据存储解决方案. MongoDB 是一个介于关系数据库和非关系数据库之间的产品,是非关系数据库当中功能最丰富,最像关系数据库的. 下面给大家介绍MongoDB查询文档操作的实例 先把student删除,再重新插入数据 > db.student.drop() true > db.student.insert([{ "_id" : 1, "

  • Mybatis和orcale update语句中接收参数为对象的实例代码

    Mybatis的 mapper.xml 中 update 语句使用 if 标签判断对像属性是否为空值. UserDTO是传过来参数的类型,userDTO是在mapperDao接口中给更新方法的参数起的别名. mapperDao.java int updata(@Param("userDTO") UserDTO userDTO); mapper.xml <update id="updata" parameterType="UserDTO"&g

  • 详解mybatis中的if-else的嵌套使用

    目录 案例一:if-else 案例二:if嵌套 MyBatis中if和choose的嵌套 案例一:if-else 在mybatis的使用过程中,难免会存在使用if-else的逻辑,但是实际是没有这种语法的,提供了choose标签来替代这种语法 <select id="selectUserByState" resultType="com.bz.model.entity.User"> SELECT * FROM user WHERE 1=1 <choo

  • Android手机通过蓝牙连接佳博打印机的实例代码

    所使用的打印机为佳博打印机,支持蓝牙.wifi.usb我所使用的是通过蓝牙来连接. 在网上找到一个佳博官方针对安卓开发的App源码,但是各种的跳转,没有看太懂,所以又去问度娘,找到了一个不错的文章 Android对于蓝牙开发从2.0版本的sdk才开始支持,而且模拟器不支持,测试至少需要两部手机,所以制约了很多技术人员的开发. 1. 首先,要操作蓝牙,先要在AndroidManifest.xml里加入权限 // 管理蓝牙设备的权限 <uses-permissionandroid:name="

随机推荐