应用场景:
从对象建模的角度来看,如果两个对象之间是一种聚集——是整体的一部分(part-of)的关系,聚集是一种强健的关联形式:它对于对象的生命周期有一些额外的语义。在这种情况下,我们有一种更强健的形式:复合(composition),在这里部分的生命周期完全依赖于整体的生命周期。对象建模专家和UML设计师声称,就实际的java实现而言,在这个复合和其他更弱形式的关联之间是没有区分的。但是在ORM的上下文种,则有一个很大德区别:复合的关系经常是一个备选的值类型。
示例实体:
User Address一个 User有一个家庭地址(homeAddress)和一个账单地址(billingAddress)。User是实体,而Address是值类型。他们是聚集关系。
一、XML的实现
package com.ccay.test.componentMapping_xml;
public class User {
private long userId;
private String firstname;
private String lastname;
private Address homeAddress;
private Address billingAddress;
public Address getHomeAddress() {
return homeAddress;
}
public void setHomeAddress(Address homeAddress) {
this.homeAddress = homeAddress;
}
public Address getBillingAddress() {
return billingAddress;
}
public void setBillingAddress(Address billingAddress) {
this.billingAddress = billingAddress;
}
public String getFirstname() {
return firstname;
}
public void setFirstname(String firstname) {
this.firstname = firstname;
}
public String getLastname() {
return lastname;
}
public void setLastname(String lastname) {
this.lastname = lastname;
}
public long getUserId() {
return userId;
}
public void setUserId(long userId) {
this.userId = userId;
}
}
package com.ccay.test.componentMapping_xml;
public class Address {
private User user;
private String street;
private String zipcode
private String city;
public User getUser() {
return user;
}
public void setUser(User user) {
this.user = user;
}
public String getStreet() {
return street;
}
public void setStreet(String street) {
this.street = street;
}
public String getZipcode() {
return zipcode;
}
public void setZipcode(String zipcode) {
this.zipcode = zipcode;
}
public String getCity() {
return city;
}
public void setCity(String city) {
this.city = city;
}
}
package com.ccay.test.componentMapping_xml;
import org.hibernate.Session;
import org.hibernate.Transaction;
import com.ccay.test.HibernateSessionFactory;
public class Test {
@org.junit.Test
public void testDDL(){
Session session = HibernateSessionFactory.getSession();
Transaction transaction = session.beginTransaction();
Address homeAddress = new Address();
homeAddress.setCity("china");
homeAddress.setStreet("zhongshan");
homeAddress.setZipcode("010");
Address billingAddress = new Address();
billingAddress.setCity("china");
billingAddress.setStreet("zhongshan2");
billingAddress.setZipcode("0102");
User user= new User();
user.setFirstname("firstName");
user.setLastname("lastName");
user.setHomeAddress(homeAddress);
user.setBillingAddress(billingAddress);
session.save(user);
transaction.commit();
HibernateSessionFactory.closeSession();
}
}
HBM.XML映射文件
<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="com.ccay.test.componentMapping_xml.User" table="COM_MAP_USER" >
<id name="userId" type="long">
<column name="USER_ID"/>
<generator class="native" />
</id>
<property name="firstname" type="string">
<column name="FIRSTNAME" length="40" />
</property>
<property name="lastname" type="string">
<column name="LASTNAME" length="50" />
</property>
<component name="homeAddress" class="com.ccay.test.componentMapping_xml.Address">
<!-- 可加入后退指针 -->
<parent name="user"/>
<property name="street" type="string" column="HOME_STREET" not-null="true"/>
<property name="city" type="string" column="HOME_CITY" not-null="true"/>
<property name="zipcode" type="string" column="HOME_ZIPCODE" not-null="true"/>
</component>
<component name="billingAddress" class="com.ccay.test.componentMapping_xml.Address">
<parent name="user"/>
<property name="street" type="string" column="BILLING_STREET" not-null="true"/>
<property name="city" type="string" column="BILLING_CITY" not-null="true"/>
<property name="zipcode" type="string" column="BILLING_ZIPCODE" not-null="true"/>
</component>
</class>
</hibernate-mapping>
hibernate自动 DDL
create table COM_MAP_USER (
USER_ID bigint not null auto_increment,
FIRSTNAME varchar(40),
LASTNAME varchar(50),
HOME_STREET varchar(255) not null,
HOME_CITY varchar(255) not null,
HOME_ZIPCODE varchar(255) not null,
BILLING_STREET varchar(255) not null,
BILLING_CITY varchar(255) not null,
BILLING_ZIPCODE varchar(255) not null,
primary key (USER_ID)
)
powerDesigner 逆向工程如图
二、注解的实现
package com.ccay.test.componentMapping_JAPAnnotation;
import javax.annotation.Generated;
import javax.persistence.AttributeOverride;
import javax.persistence.AttributeOverrides;
import javax.persistence.Column;
import javax.persistence.Embedded;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.Id;
import javax.persistence.Table;
@Entity
@Table(name = "COM_MAP_USER")
public class User {
@Id
@GeneratedValue
private long userId;
private String firstname;
private String lastname;
@Embedded
@AttributeOverrides({//可以覆盖属性值,重命名列名
@AttributeOverride(name="street",column = @Column(name="HOME_STREET")),
@AttributeOverride(name="zipcode",column = @Column(name="HOME_ZIPCODE")),
@AttributeOverride(name="city",column = @Column(name="HOME_CITY"))
})
private Address homeAddress;
@Embedded
private Address billingAddress;
public Address getHomeAddress() {
return homeAddress;
}
public void setHomeAddress(Address homeAddress) {
this.homeAddress = homeAddress;
}
public Address getBillingAddress() {
return billingAddress;
}
public void setBillingAddress(Address billingAddress) {
this.billingAddress = billingAddress;
}
public String getFirstname() {
return firstname;
}
public void setFirstname(String firstname) {
this.firstname = firstname;
}
public String getLastname() {
return lastname;
}
public void setLastname(String lastname) {
this.lastname = lastname;
}
public long getUserId() {
return userId;
}
public void setUserId(long userId) {
this.userId = userId;
}
}
package com.ccay.test.componentMapping_JAPAnnotation;
import javax.persistence.Column;
import javax.persistence.Embeddable;
import org.hibernate.annotations.Parent;
@Embeddable
public class Address {
@Parent//JPA中没有 所以用 org.hibernate.annotations.Parent
private User user;
@Column(name="ADDRESS_STREET",nullable=false)
private String street;
@Column(name="ADDRESS_ZIPCODE",nullable=false)
private String zipcode;
@Column(name="ADDRESS_CITY",nullable=false)
private String city;
public User getUser() {
return user;
}
public void setUser(User user) {
this.user = user;
}
public String getStreet() {
return street;
}
public void setStreet(String street) {
this.street = street;
}
public String getZipcode() {
return zipcode;
}
public void setZipcode(String zipcode) {
this.zipcode = zipcode;
}
public String getCity() {
return city;
}
public void setCity(String city) {
this.city = city;
}
}
hibernate 自动DDL
create table COM_MAP_USER (
userId bigint not null auto_increment,
ADDRESS_CITY varchar(255) not null,
ADDRESS_STREET varchar(255) not null,
ADDRESS_ZIPCODE varchar(255) not null,
firstname varchar(255),
HOME_CITY varchar(255),
HOME_STREET varchar(255),
HOME_ZIPCODE varchar(255),
lastname varchar(255),
primary key (userId)
)
逆向工程