Mybatis多表关联查询的实现(DEMO)
概要
本节要实现的是多表关联查询的简单demo。场景是根据id查询某商品分类信息,并展示该分类下的商品列表。
一、mysql测试数据
新建表category(商品分类)和product(商品),并插入几条测试数据。
create table category ( id int not null auto_increment, name varchar(80) null, constraint pk_category primary key (id) ); insert into category(name) values ('女装'); insert into category(name) values ('美妆'); insert into category(name) values ('书籍'); create table product ( id int not null auto_increment, categoryid int not null, name varchar(80) null, constraint pk_product primary key (id), constraint fk_product_2 foreign key (categoryid) references category (id) ); create index productcat on product (categoryid); create index productname on product (name); insert into product(categoryid,name) values (1, '裂帛'); insert into product(categoryid,name) values (1, '雅鹿'); insert into product(categoryid,name) values (2,'膜法世家'); insert into product(categoryid,name) values (2,'御泥坊'); insert into product(categoryid,name) values (2, '雅诗兰黛'); insert into product(categoryid,name) values (2, '欧莱雅'); insert into product(categoryid,name) values (2, '韩后'); insert into product(categoryid,name) values (2, '相宜本草'); insert into product(categoryid,name) values (3,'疯狂java'); insert into product(categoryid,name) values (3,'java核心技术');
二、配置mybatis-generator-config.xml
配置mybatis-generator-config.xml的方法见 java入门[7]-mybatis generator(mbg)自动生成mybatis代码 ,这里主要改动的是table节点。
<table tablename="category" enablecountbyexample="true" enabledeletebyexample="true" enableselectbyexample="true" enableupdatebyexample="true"> <generatedkey column="id" sqlstatement="mysql" identity="true"/> </table> <table tablename="product" enablecountbyexample="true" enableselectbyexample="true" enableselectbyprimarykey="true" enableupdatebyprimarykey="true" enabledeletebyprimarykey="true" enableinsert="true"> <generatedkey column="id" sqlstatement="mysql" identity="true"></generatedkey> </table>
配置好xml文件后,在maven面板运行mybatis-generator:generate,自动生成相关的类。
三、自定义mybatis关联查询
1.封装实体dto
我们新定义categorydto,封装商品分类信息及其商品列表。
public class categorydto { private category category; private list<product> products; private int id; public int getid() { return id; } public void setid(int id) { this.id = id; } public category getcategory() { return category; } public void setcategory(category category) { this.category = category; } public list<product> getproducts() { return products; } public void setproducts(list<product> products) { this.products = products; } }
2.为categorymapper.java接口新增方法getbyid()
categorydto getbyid(int id);
3.配置categorymapper.xml
首先定义select节点,id对应上面的方法名getbyid;parametertype参数类型为integer;resultmap为自定义resultmap的id。
<select id="getbyid" parametertype="java.lang.integer" resultmap="categoryresult"> select category.id as cateid,category.name as catename,product.id as productid,product.name as productname from category,product where category.id=product.categoryid and category.id=#{id} </select>
接下来定义resultmap节点id为categoryresult,type为categorydto。
关于resultmap:
- id – 一个 id 结果;标记结果作为 id 可以帮助提高整体效能
- result – 注入到字段或 javabean 属性的普通结果
- association – 一个复杂的类型关联;许多结果将包成这种类型
- 嵌入结果映射 – 结果映射自身的关联,或者参考一个
- collection – 复杂类型的集
- 嵌入结果映射 – 结果映射自身的集,或者参考一个
完整参考官网:http://www.mybatis.org/mybatis-3/zh/sqlmap-xml.html#result_maps
用association对应category,collection对应products,然后用result对应到每个具体字段。
<resultmap id="categoryresult" type="com.data.dto.categorydto"> <association property="category" javatype="com.data.pojo.category"> <result property="id" column="cateid"></result> <result property="name" column="catename"></result> </association> <collection property="products" oftype="com.data.pojo.product"> <result property="id" column="productid"></result> <result property="name" column="productname"></result> </collection> </resultmap>
四、测试
在上一节测试基础上新增测试方法:
@test public void test_getbyid(){ int id=2; categorydto dto= categorymapper.getbyid(id); if(dto==null){ system.out.println("不存在"); }else { system.out.println("商品id="+dto.getid()+" name="+dto.getcategory().getname()); system.out.println("products:"+dto.getproducts().size()); for(product product:dto.getproducts()){ system.out.println(" |_"+product.getname()); } } }
运行之后居然报错了
org.mybatis.spring.mybatissystemexception: nested exception is org.apache.ibatis.exceptions.toomanyresultsexception: expected one result (or null) to be returned by selectone(), but found: 6
后来找到了解决方案,修改resultmap,添加id节点就可以了。
<resultmap id="categoryresult" type="com.data.dto.categorydto"> <id property="id" column="cateid"></id> …… </resultmap>
运行结果:
商品id=2 name=美妆
products:6
|_膜法世家
|_御泥坊
|_雅诗兰黛
|_欧莱雅
|_韩后
|_相宜本草