实际的开发中,对数据库的操作常常会涉及到多张表,这在面向对象中就涉及到了对象与对象之间的关联关系。针对多表之间的操作,MyBatis提供了关联映射,通过关联映射就可以很好的处理对象与对象之间的关联关系。
在关系型数据库中,多表之间存在着三种关系,分别是一对一,一对多,多对多。
一对一:在任意一个表中引入另外一个表的主键作为外键。
一对多:在多个表中都引入了某一个表的主键作为外键
多对多:需要用一张中间表表示多对多的关系,这张中间表引入两张表的主键作为外键。
一般来说一个对象映射一张表,因此一对一的关系就是在A类中定义B类属性,一对多的关系就是在A类中定义List< B> 的属性,多对多就是分别在A、B类中定义对方的List 属性。
一对一关系是一个基本的映射关系,比如Person(人)--IDCard(身份证),我们可以通过如下两种方式实现:
配置映射文件来实现一对一的映射关系,实现级联查询,要求通过person可以获取到对应的idencard信息
关于级联查询:若表A中有一个外键引用了表B的主键,A表就是子表,B表就是父表。当查询表A的数据时,通过表A的外键将表B的记录也查找出来,这就是级联查询。相应的还有级联删除,当删除B表的记录时,会先将A表中关联的记录删掉
(1)person表和 idencard表
-- 创建 idencard 表
-- 记录身份证
CREATE TABLE `idencard`(
`id` INT PRIMARY KEY AUTO_INCREMENT,
`card_sn` VARCHAR(32) NOT NULL DEFAULT ''
)CHARSET utf8;
INSERT INTO `idencard` VALUES(1,'123456789098765');
-- 创建person表
CREATE TABLE `person`(
`id` INT PRIMARY KEY AUTO_INCREMENT,
`name` VARCHAR(32) NOT NULL DEFAULT '',
`card_id` INT, -- 对应idencard表的主键-id
FOREIGN KEY (`card_id`) REFERENCES idencard (`id`)-- card_id作为外键
)CHARSET utf8;
INSERT INTO `person` VALUES(1,'张三',1);
(2)实体类 IdenCard 和 Person
package com.li.entity;
/**
* @author 李
* @version 1.0
*/
public class IdenCard {
private Integer id;
private String card_sn;
//省略setter,getter,toString方法
}
package com.li.entity;
/**
* @author 李
* @version 1.0
*/
public class Person {
private Integer id;
private String name;
private IdenCard card;
//省略setter,getter,toString方法
}
(3)PersonMapper 接口
package com.li.mapper;
import com.li.entity.Person;
/**
* @author 李
* @version 1.0
*/
public interface PersonMapper {
//通过Person的id获取到Person,包括这个Person关联的IdenCard对象(级联操作)
public Person getPersonById(Integer id);
}
PersonMapper.xml映射文件实现级联查询,实现方法是使用多表联查,返回的数据通过resultMap结果映射
<mapper namespace="com.li.mapper.PersonMapper">
<!--1.接口声明:public Person getPersonById(Integer id);
2.通过Person的id获取到Person,包括这个Person关联的IdenCard对象(级联操作)
3.返回类型如果配置成resultType="Person",不能实现级联查询,
在返回的person对象中IdenCard属性对象为 null
4.因此需要使用自定义resultMap,在resultMap中指定级联关系-->
<select id="getPersonById" parameterType="Integer" resultMap="PersonResultMap">
SELECT * FROM `person`,`idencard` WHERE `person`.`id`= #{id} AND
`person`.`card_id` = `idencard`.`id`;
</select>
<!--association – 一个复杂类型的关联;许多结果将包装成这种类型嵌套结果映射 – 关联可以是
resultMap 元素,或是对其它结果映射的引用
1.property="card" 表示 Person对象的card属性
2.javaType="IdenCard" 表示card属性的类型-->
<resultMap id="PersonResultMap" type="Person">
<!--id标签–一个ID结果(就是主键);标记出作为主键的结果可以帮助提高整体性能-->
<!--这里的property表示Person类的属性名,column表示对应表的字段-->
<id property="id" column="id"/>
<result property="name" column="name"/>
<association property="card" javaType="IdenCard">
<!--这里的property表示IdenCard类的属性名,column表示表的字段名-->
<result property="id" column="id"/>
<result property="card_sn" column="card_sn"/>
</association>
</resultMap>
</mapper>
测试:
@Test
public void getPersonById() {
Person person = personMapper.getPersonById(1);
System.out.println("person=" + person);
if (sqlSession != null) {
sqlSession.close();
}
}
第一种方式使用了多表联查的形式实现级联查询,但是如果涉及的表过多,sql语句可读性就会变差。第二种方式的核心思想是将多表联查分解成单表操作,这样更简洁,易于维护,而且可以复用你写好的方法,推荐使用
(1)创建IdenCardMapper接口
public interface IdenCardMapper {
//根据id获取到身份证序列号
public IdenCard getIdenCardById(Integer id);
}
(2)在IdenCardMapper的映射文件中实现该方法
<mapper namespace="com.li.mapper.IdenCardMapper">
<!--配置实现public IdenCard getIdenCardById(Integer id);-->
<select id="getIdenCardById" parameterType="Integer" resultType="IdenCard">
SELECT * FROM `idencard` WHERE `id` = #{id}
</select>
</mapper>
(3)PersonMapper接口
//通过Person的id获取到Person,包括这个Person关联的IdenCard对象(方式2)
public Person getPersonById2(Integer id);
(4)实现PersonMapper接口的映射文件
1. 先通过 SELECT * FROM person WHERE id =#{id} 返回 person 信息
2. 以第一个操作返回的 card_id 字段数据,作为条件再次查询,得到对应的 IdenCard 数据
如果第一个操作使用了别名,那么返回的时候的字段也是别名,因此第二个操作也要使用别名才能匹配到
<!--通过Person的id获取到Person,包括这个Person关联的IdenCard对象(方式2)
接口方法:public Person getPersonById2(Integer id);-->
<resultMap id="PersonResultMap2" type="Person">
<id property="id" column="id"/>
<result property="name" column="name"/>
<!--第二种方式的核心思想是将多表联查操作分解成单表操作,
这样更简洁,易于维护,复用性更强,推荐使用-->
<!--1.property="card"表示Person对象的card属性
2.column="card_id"是SELECT * FROM person WHERE id = #{id}语句返回的card_id字段名/别名
3.返回的字段card_id信息/数据会作为getIdenCardById()的入参来执行方法-->
<association property="card" column="card_id"
select="com.li.mapper.IdenCardMapper.getIdenCardById"/>
</resultMap>
<select id="getPersonById2" parameterType="Integer" resultMap="PersonResultMap2">
SELECT * FROM person WHERE id = #{id};
</select>
测试结果:
可以看到底层执行了两次sql查询操作。首先对person表进行查询,查询结果(card_id)作为第二张表的查询条件(id),再对idencard表进行查询。
通过注解的方式来实现一对一的映射关系,实现级联查询,通过person可以获取到对应的idencard的信息。这里只进行方式二的演示。
在实际开发中还是推荐使用配置方式
(1)注解实现方法
IdenCardMapperAnnotation 接口:
package com.li.mapper;
import com.li.entity.IdenCard;
import org.apache.ibatis.annotations.Select;
/**
* @author 李
* @version 1.0
* 使用注解的方式实现一对一的映射
*/
public interface IdenCardMapperAnnotation {
//根据id获取到身份证序列号
@Select(value = "SELECT * FROM `idencard` WHERE `id` = #{id}")
public IdenCard getIdenCardById(Integer id);
}
PersonMapperAnnotation 接口:
package com.li.mapper;
import com.li.entity.Person;
import org.apache.ibatis.annotations.One;
import org.apache.ibatis.annotations.Result;
import org.apache.ibatis.annotations.Results;
import org.apache.ibatis.annotations.Select;
/**
* @author 李
* @version 1.0
*/
public interface PersonMapperAnnotation {
//通过Person的id获取到Person,包括这个Person关联的IdenCard对象
//注解的形式就是对前面xml配置方式的体现
@Select(value = "SELECT * FROM person WHERE id = #{id}")//如果这里返回的字段使用了别名,则@result的card_id也要使用该别名
@Results({//配置返回数据的映射
@Result(id = true, property = "id", column = "id"),
@Result(property = "name", column = "name"),
@Result(property = "card", column = "card_id",
one = @One(select = "com.li.mapper.IdenCardMapper.getIdenCardById"))
})
public Person getPersonById(Integer id);
}
(2)测试
@Test
public void getIdenCardById() {
Person person = personMapperAnnotation.getPersonById(1);
System.out.println("person=" + person);
if (sqlSession != null) {
sqlSession.close();
}
}
一张表是否设置了外键,对MyBatis进行对象级联映射没有影响,外键只是对表本身数据的约束
前面我们讲解的是查询Person可以级联查询到IdenCard,如果要求通过查询IdenCard,也可以级联查询到Person,应该如何解决?
这实际上是Person和IdenCard的1对1的双向映射(1<=>1),解决办法除了传统的多表联查之外,仍然可以使用分解为多次单表操作的方式:
1. 先通过 SELECT * FROM idencard WHERE id =#{id} 返回 IdenCard 信息
2. 以第一个操作返回的 id 字段数据,作为条件再次查询,得到对应的 person数据
3. 将结果进行映射,放在resultMap中
(1)修改IdenCard实体类,添加Person属性
package com.li.entity;
public class IdenCard {
private Integer id;
private String card_sn;
private Person person;
//省略setter、getter、toString方法
}
Person实体类不变。
(2)IdenCard接口
//练习-根据id获取到身份证序列号,同时返回对应的Person信息
public IdenCard getIdenCardById2(Integer cardId);
(3)Person接口
//通过Person的card_id获取到Person信息
public Person getPersonByCardId(Integer cardId);
(4)IdenCardMapper.xml
<resultMap id="IdenCardMap" type="IdenCard">
<id property="id" column="id"/>
<result property="card_sn" column="card_sn"/>
<!--1.property="person" 是指 IdenCard类的person属性名
2.column="id" 是指getIdenCardById2返回的id字段名作为第二次查询的card_id-->
<association property="person" column="id"
select="com.li.mapper.PersonMapper.getPersonByCardId"/>
</resultMap>
<!--练习-根据id获取到身份证序列号,同时返回对应的Person信息
public IdenCard getIdenCardById2(Integer id);-->
<select id="getIdenCardById2" parameterType="Integer" resultMap="IdenCardMap">
SELECT * FROM `idencard` WHERE `id` = #{id}
</select>
(5)PersonMapper.xml
<!--练习-通过Person的card_id获取到Person-->
<select id="getPersonByCardId" parameterType="Integer" resultType="Person">
SELECT * FROM person WHERE card_id = #{cardId};
</select>
(6)测试
@Test
public void getIdenCardById2() {
IdenCard idenCard = idenCardMapper.getIdenCardById2(1);
System.out.println("idenCard=" + idenCard);
}
我正在使用Sequel构建一个愿望list系统。我有一个wishlists和itemstable和一个items_wishlists连接表(该名称是续集选择的名称)。items_wishlists表还有一个用于facebookid的额外列(因此我可以存储opengraph操作),这是一个NOTNULL列。我还有Wishlist和Item具有续集many_to_many关联的模型已建立。Wishlist类也有:selectmany_to_many关联的选项设置为select:[:items.*,:items_wishlists__facebook_action_id].有没有一种方法可以
我的问题的一个例子是体育游戏。一场体育比赛有两支球队,一支主队和一支客队。我的事件记录模型如下:classTeam"Team"has_one:away_team,:class_name=>"Team"end我希望能够通过游戏访问一个团队,例如:Game.find(1).home_team但我收到一个单元化常量错误:Game::team。谁能告诉我我做错了什么?谢谢, 最佳答案 如果Gamehas_one:team那么Rails假设您的teams表有一个game_id列。不过,您想要的是games表有一个team_id列,在这种情况下
目前,Itembelongs_toCompany和has_manyItemVariants。我正在尝试使用嵌套的fields_for通过Item表单添加ItemVariant字段,但是使用:item_variants不显示该表单。只有当我使用单数时才会显示。我检查了我的关联,它们似乎是正确的,这可能与嵌套在公司下的项目有关,还是我遗漏了其他东西?提前致谢。注意:下面的代码片段中省略了不相关的代码。编辑:不知道这是否相关,但我正在使用CanCan进行身份验证。routes.rbresources:companiesdoresources:itemsenditem.rbclassItemi
我认为我的问题最好用一个例子来描述。假设我有一个名为“Thing”的简单模型,它有一些简单数据类型的属性。像...Thing-foo:string-goo:string-bar:int这并不难。数据库表将包含具有这三个属性的三列,我可以使用@thing.foo或@thing.bar之类的东西访问它们。但我要解决的问题是当“foo”或“goo”不再包含在简单数据类型中时会发生什么?假设foo和goo代表相同类型的对象。也就是说,它们都是“Whazit”的实例,只是数据不同。所以现在事情可能看起来像这样......Thing-bar:int但是现在有一个新的模型叫做“Whazit”,看起来
我想获取主题名称与搜索关键字匹配的所有配置文件。现在我正在加载所有配置文件。我需要知道如何实现它。非常感谢任何帮助。配置文件.rbhas_many:categorizationshas_many:subjects,through::categorizations主题.rbhas_many:categorizationshas_many:profiles,through::categorizations分类.rbbelongs_to:profilebelongs_to:subjectviews/search/index.html.erb#searchform'get'do%>nil%>#
如果names为nil,则以下中断。我怎样才能让这个map只有在它不是nil时才执行?self.topics=names.split(",").mapdo|n|Topic.where(name:n.strip).first_or_create!end 最佳答案 其他几个选项:选项1(在其上执行map时检查split的结果):names_list=names.try(:split,",")self.topics=names_list.mapdo|n|Topic.where(name:n.strip).first_or_create!e
我有一个非常简单的Controller来管理我的Rails应用程序中的静态页面:classPagesController我怎样才能让View模板返回它自己的名字,这样我就可以做这样的事情:#pricing.html.erb#-->"Pricing"感谢您的帮助。 最佳答案 4.3RoutingParametersTheparamshashwillalwayscontainthe:controllerand:actionkeys,butyoushouldusethemethodscontroller_nameandaction_nam
我使用的是遗留数据库,所以我无法控制数据模型。他们使用了很多多态链接/连接表,就像这样createtableperson(per_ident,name,...)createtableperson_links(per_ident,obj_name,obj_r_ident)createtablereport(rep_ident,name,...)其中obj_name是表名,obj_r_ident是标识符。因此链接的报告将按如下方式插入:insertintoperson(1,...)insertintoreport(1,...)insertintoreport(2,...)insertint
我遇到了RVM的问题,所以我卸载并重新安装了它。事实是我实际上尝试过rbenv,但这对我来说没有用,所以我试图让rvm重新启动并运行-而不必安装重复版本的Ruby。我至少安装了1个现有版本的Ruby:ruby--versionruby1.8.7(2011-12-28patchlevel357)[universal-darwin11.0]但是当我执行rvmlist时,我得到一个空白列表:bash-3.2$rvmlistrvmrubies#Defaultrubynotset.Try'rvmaliascreatedefault'.#=>-current#=*-current&&default
在我的Rails项目中,我有三个模型:classRecipe:recipe_categorizationsaccepts_nested_attributes_for:recipe_categories,allow_destroy::trueendclassCategory:recipe_categorizationsendclassRecipeCategorization通过这个简单的has_many:through设置,我怎样才能像这样获取给定的食谱:@recipe=Recipe.first并根据现有类别向此食谱添加类别,并在相应类别上对其进行更新。所以:@category=#Exi