一对一关联分为主键关联与外键关联。
主键关联:不必加额外的字段,只是主表和辅表的主键相关联,即这两个主键的值是一样的。
外键关联:辅表有一个额外的字段和主表相关联,或者两个表都有额外的字段与对应表的相关联。
官方文档解释
<one-to-one
name="propertyName"
(1)
class="ClassName"
(2)
cascade="cascade_style"
(3)
constrained="true|false"
(4)
fetch="join|select"
(5)
property-ref="propertyNameFromAssociatedClass"
(6)
access="field|property|ClassName"
(7)
formula="any SQL expression"
(8)
lazy="proxy|no-proxy|false"
(9)
entity-name="EntityName"
(10)
node="element-name|@attribute-name|element/@attribute|."
embed-xml="true|false"
foreign-key="foreign_key_name"
/>
(1) name: 属性的名字。
(2) class (可选 - 默认是通过反射得到的属性类型):被关联的类的名字。
(3) cascade(级联) (可选):表明操作是否从父对象级联到被关联的对象。
(4) constrained(约束) (可选):表明该类对应的表对应的数据库表,和被关联的对象所对应的数据库表之间,通过一个外键引用对主键进行约束。 这个选项影响save()和delete()在级联执行时的先后顺序以及 决定该关联能否被委托(也在schema export tool中被使用).
(5) fetch (可选 - 默认设置为选择): 在外连接抓取或者序列选择抓取选择其一.
(6) property-ref (可选):指定关联类的属性名,这个属性将会和本类的主键相对应。如果没有指定,会使用对方关联类的主键。
(7) access (可选 - 默认是 property): Hibernate用来访问属性的策略。
(8) formula (可选):绝大多数一对一的关联都指向其实体的主键。在一些少见的情况中, 你可能会指向其他的一个或多个字段,或者是一个表达式,这些情况下,你可以用一个SQL公式来表示。 (可以在org.hibernate.test.onetooneformula找到例子)
(9) lazy (可选 - 默认为 proxy): 默认情况下,单点关联是经过代理的。lazy="no-proxy"指定此属性应该在实例变量第一次被访问时应该延迟抓取(fetche lazily)(需要运行时字节码的增强)。 lazy="false"指定此关联总是被预先抓取。注意,如果constrained="false", 不可能使用代理,Hibernate会采取预先抓取!
(10) entity-name (可选): 被关联的类的实体名。
1. 主键关联
唯一外键关联:
例: User和Account,一个用户对应一个账户。
配置文件都用one-to-one,在辅表的one-to-one的属性里要加constrained="true"表示受到约束。所以,将辅表的id改成foreign然后加上属性参数等。
User
Account
User类与配置文件
private Integer id;
private String username;
private String password;
private Account account;
one-to-one中的class属性可以不写,默认Hibernate会使用反正自己去寻找。
<one-to-one name="account" class="piaohan.domain.Account"cascade="all" />
Account类与配置
private Integer id;
private String accountNum;
private Integer money;
private User user;
<id name="id" column="id">
<generator class="foreign">
<param name="property">user</param>
</generator>
</id>
<one-to-one name="user" constrained="true"
class="piaohan.domain.User">
</one-to-one>
constrained="true"表示这个属性(name="user")对应类型的表(user表),对我现在的表(account表)形成了外键约束,user表的主键对account表的主键形成了外键约束。
该种方式如果删除想删除user对象中的account对象,不能直接删除account,需要先删除双方关系,如user.setAccount(null);或者在account方加上cascade。
2. 单方外键关联
例: User和Account,一个用户对应一个账户。
User
Account
这种关联Account是用many-to-one然后用unique="true"做限制,限制成一对一的关系。所以,一对一的外键关联其实就是多对一关联的一种特例。
User类与配置
private Integer id;
private String username;
private String password;
private Account account;
<one-to-one name="account" class="piaohan.domain.Account"
cascade="all" property-ref="user" />
property-ref="user" 表示关联类的属性名,即对应类的保存本类的属性名,该例在User中,表示对应类Account中,保存User属性的属性名,即user。
Account类与配置
private Integer id;
private String accountNum;
private Integer money;
private User user;
<many-to-one name="user" unique="true" column="userid" />
3. 双方外键关联
如果关联双方都是外键关联,那可以两边都用many-to-one。
User配置(类不变)
<many-to-one name="account" cascade="all" unique="true"
column="accountid" />
Account配置(类不变)
<many-to-one name="user" unique="true" column="userid"
cascade="all" />
1、主键关联
User
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
public Integer getId() {
return id;
}
@OneToOne(cascade = CascadeType.ALL, mappedBy = "user")
public Account getAccount() {
return account;
}
Account
@OneToOne(cascade = CascadeType.ALL)
@PrimaryKeyJoinColumn
public User getUser() {
return user;
}
@Id
@GeneratedValue(generator = "for")
@GenericGenerator(name = "for", strategy = "foreign",
parameters = { @Parameter(name = "property", value ="user") })
public Integer getId() {
return id;
}
注释方式与配置文件方式实现相同,只不过配置方式有所不同而已。
2. 单方外键关联
User
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
public Integer getId() {
return id;
}
@OneToOne(cascade = CascadeType.ALL, mappedBy = "user")
public Account getAccount() {
return account;
}
Account
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
public Integer getId() {
return id;
}
@ManyToOne(cascade = CascadeType.ALL, unique = true)
@JoinColumn(name = "userid")
public User getUser() {
return user;
}
3. 双方外键关联
User类配置
@OneToOne(cascade = CascadeType.ALL)
@JoinColumn(name = "accountid", unique = true)
public Account getAccount() {
return account;
}
Account配置(类不变)
@OneToOne(cascade = CascadeType.ALL)
@JoinColumn(name = "userid", unique = true)
public User getUser() {
return user;
}
三、备注
一对一映射还有其他方式,如使用中间表等,这里只讨论两表关联,中间表就是多对多的一种特例,可以在多对多时设置唯一约束即可,虽然这种方式极少被使用。