Hibernate Annotations
Hibernate Annotations
Hibernate Annotations
Reference Guide
Version: 3.3.0.GA
Table of Contents
Preface ............................................................................................................................................ iv
1. Setting up an annotations project ................................................................................................ 1
1.1. Requirements ..................................................................................................................... 1
1.2. Configuration ..................................................................................................................... 1
1.3. Properties .......................................................................................................................... 2
2. Entity Beans ................................................................................................................................ 4
2.1. Intro .................................................................................................................................. 4
2.2. Mapping with EJB3/JPA Annotations .................................................................................. 4
2.2.1. Declaring an entity bean .......................................................................................... 4
2.2.1.1. Defining the table ......................................................................................... 4
2.2.1.2. Versioning for optimistic locking ................................................................... 5
2.2.2. Mapping simple properties ....................................................................................... 5
2.2.2.1. Declaring basic property mappings ................................................................ 5
2.2.2.2. Declaring column attributes ........................................................................... 7
2.2.2.3. Embedded objects (aka components) .............................................................. 7
2.2.2.4. Non-annotated property defaults .................................................................... 9
2.2.. Mapping identifier properties ..................................................................................... 9
2.2.4. Mapping inheritance .............................................................................................. 12
2.2.4.1. Table per class ............................................................................................ 12
2.2.4.2. Single table per class hierarchy .................................................................... 13
2.2.4.3. Joined subclasses ........................................................................................ 13
2.2.4.4. Inherit properties from superclasses ............................................................. 14
2.2.5. Mapping entity bean associations/relationships ........................................................ 15
2.2.5.1. One-to-one ................................................................................................. 15
2.2.5.2. Many-to-one ............................................................................................... 17
2.2.5.3. Collections ................................................................................................. 18
2.2.5.4. Transitive persistence with cascading ........................................................... 23
2.2.5.5. Association fetching ................................................................................... 24
2.2.6. Mapping composite primary and foreign keys ......................................................... 24
2.2.7. Mapping secondary tables ...................................................................................... 25
2.3. Mapping Queries .............................................................................................................. 26
2.3.Mapping JPAQL/HQL queries. Mapping JPAQL/HQL queries .................................... 26
2.3.2. Mapping native queries .......................................................................................... 27
2.4. Hibernate Annotation Extensions ...................................................................................... 30
2.4.1. Entity ................................................................................................................... 30
2.4.Identifier. Identifier ................................................................................................... 32
2.4.3. Property ................................................................................................................ 32
2.4.3.1. Access type ................................................................................................ 33
2.4.3.2. Formula ..................................................................................................... 34
2.4.3.3. Type .......................................................................................................... 34
2.4.3.4. Index ......................................................................................................... 35
2.4.3.5. @Parent ..................................................................................................... 35
2.4.3.6. Generated properties ................................................................................... 35
2.4.3.7. @Target ..................................................................................................... 36
2.4.3.8. Optimistic lock ........................................................................................... 36
2.4.4. Inheritance ............................................................................................................ 36
2.4.5. Single Association related annotations .................................................................... 36
2.4.5.Lazy options and fetching modes. Lazy options and fetching modes .................. 37
Hibernate 3.3.0.GA ii
Hibernate Annotations
The EJB3 specification recognizes the interest and the success of the transparent object/relational mapping
paradigm. The EJB3 specification standardizes the basic APIs and the metadata needed for any object/relational
persistence mechanism. Hibernate EntityManager implements the programming interfaces and lifecycle rules
as defined by the EJB3 persistence specification. Together with Hibernate Annotations, this wrapper imple-
ments a complete (and standalone) EJB3 persistence solution on top of the mature Hibernate core. You may use
a combination of all three together, annotations without EJB3 programming interfaces and lifecycle, or even
pure native Hibernate, depending on the business and technical needs of your project. You can at all times fall
back to Hibernate native APIs, or if required, even to native JDBC and SQL.
This release is based on the final release of the EJB 3.0 / JPA specification (aka JSP-220) and support all the
specification features (including the optional ones). Most of the Hibernate features and extensions are also
available through Hibernate specific annotations compared to the specification are also available. While the Hi-
bernate feature coverage is now very high, some are still missing. The eventual goal is to cover all of them. See
the JIRA road map section for more informations.
If you are moving from previous Hibernate Annotations versions, please have a look at Java Persistence migra-
tion guide [http://www.hibernate.org/398.html].
Hibernate 3.3.0.GA iv
Chapter 1. Setting up an annotations project
1.1. Requirements
• Download and unpack the Hibernate Annotations distribution from the Hibernate website.
• Make sure you have JDK 5.0 installed or above. You can of course continue using XDoclet and get some of
the benefits of annotation-based metadata with older JDK versions. Note that this document only describes
JDK 5.0 annotations and you have to refer to the XDoclet documentation for more information.
1.2. Configuration
First, set up your classpath (after you have created a new project in your favorite IDE):
• Copy all Hibernate3 core and required 3rd party library files (see lib/README.txt in Hibernate).
If you wish to use Hibernate Validator (TODO make a link to HV doc), download it from the Hibernate website
and add hibernate-validator.jar in your classpath.
If you wish to use Hibernate Search (TODO make a link to HSearch doc), download it from the Hibernate web-
site and add hibernate-search.jar and lucene-core-x.y.z.jar in your classpath.
We also recommend a small wrapper class to startup Hibernate in a static initializer block, known as Hibern-
ateUtil. You might have seen this class in various forms in other areas of the Hibernate documentation. For
Annotation support you have to enhance this helper class as follows:
package hello;
import org.hibernate.*;
import org.hibernate.cfg.*;
import test.*;
import test.animals.Dog;
static {
try {
Hibernate 3.3.0.GA 1
Setting up an annotations project
return sessionFactory.openSession();
}
}
Interesting here is the use of AnnotationConfiguration. The packages and annotated classes are declared in
your regular XML configuration file (usually hibernate.cfg.xml). Here is the equivalent of the above declara-
tion:
<hibernate-configuration>
<session-factory>
<mapping package="test.animals"/>
<mapping class="test.Flight"/>
<mapping class="test.Sky"/>
<mapping class="test.Person"/>
<mapping class="test.animals.Dog"/>
<mapping resource="test/animals/orm.xml"/>
</session-factory>
</hibernate-configuration>
Note that you can mix the hbm.xml use and the new annotation one. The resource element can be either an hbm
file or an EJB3 XML deployment descriptor. The distinction is transparent for your configuration process.
Alternatively, you can define the annotated classes and packages using the programmatic API
You can also use the Hibernate EntityManager which has its own configuration mechanism. Please refer to this
project documentation for more details.
There is no other difference in the way you use Hibernate APIs with annotations, except for this startup routine
change or in the configuration file. You can use your favorite configuration method for other properties ( hi-
bernate.properties, hibernate.cfg.xml, programmatic APIs, etc). You can even mix annotated persistent
classes and classic hbm.cfg.xml declarations with the same SessionFactory. You can however not declare a
class several times (whether annotated or through hbm.xml). You cannot mix configuration strategies (hbm vs
annotations) in a mapped entity hierarchy either.
To ease the migration process from hbm files to annotations, the configuration mechanism detects the mapping
duplication between annotations and hbm files. HBM files are then prioritized over annotated metadata on a
class to class basis. You can change the priority using hibernate.mapping.precedence property. The default is
hbm, class, changing it to class, hbm will prioritize the annotated classes over hbm files when a conflict oc-
curs.
1.3. Properties
Asides from the Hibernate core properties, Hibernate Annotations reacts to the following one
Hibernate 3.3.0.GA 2
Setting up an annotations project
Property Definition
Hibernate 3.3.0.GA 3
Chapter 2. Entity Beans
2.1. Intro
This section covers EJB 3.0 (aka Java Persistence) entity annotations and Hibernate-specific extensions.
EJB3 annotations are in the javax.persistence.* package. Most JDK 5 compliant IDE (like Eclipse, IntelliJ
IDEA and Netbeans) can autocomplete annotation interfaces and attributes for you (even without a specific
"EJB3" module, since EJB3 annotations are plain JDK 5 annotations).
For more and runnable concrete examples read the JBoss EJB 3.0 tutorial or review the Hibernate Annotations
test suite. Most of the unit tests have been designed to represent a concrete example and be a inspiration source.
Every bound persistent POJO class is an entity bean and is declared using the @Entity annotation (at the class
level):
@Entity
public class Flight implements Serializable {
Long id;
@Id
public Long getId() { return id; }
@Entity declares the class as an entity bean (i.e. a persistent POJO class), @Id declares the identifier property of
this entity bean. The other mapping declarations are implicit. This configuration by exception concept is central
to the new EJB3 specification and a major improvement. The class Flight is mapped to the Flight table, using
the column id as its primary key column.
Depending on whether you annotate fields or methods, the access type used by Hibernate will be field or
property. The EJB3 spec requires that you declare annotations on the element type that will be accessed, i.e.
the getter method if you use property access, the field if you use field access. Mixing EJB3 annotations in
both fields and methods should be avoided. Hibernate will guess the access type from the position of @Id or
@EmbeddedId.
Hibernate 3.3.0.GA 4
Entity Beans
@Tableis set at the class level; it allows you to define the table, catalog, and schema names for your entity bean
mapping. If no @Table is defined the default values are used: the unqualified class name of the entity.
@Entity
@Table(name="tbl_sky")
public class Sky implements Serializable {
...
The @Table element also contains a schema and a catalog attributes, if they need to be defined. You can also
define unique constraints to the table using the @UniqueConstraint annotation in conjunction with @Table (for
a unique constraint bound to a single column, refer to @Column).
@Table(name="tbl_sky",
uniqueConstraints = {@UniqueConstraint(columnNames={"month", "day"})}
)
A unique constraint is applied to the tuple month, day. Note that the columnNames array refers to the logical
column names.
The logical column name is defined by the Hibernate NamingStrategy implementation. The default EJB3 nam-
ing strategy use the physical column name as the logical column name. Note that this may be different than the
property name (if the column name is explicit). Unless you override the NamingStrategy, you shouldn't worry
about that.
You can add optimistic locking capability to an entity bean using the @Version annotation:
@Entity
public class Flight implements Serializable {
...
@Version
@Column(name="OPTLOCK")
public Integer getVersion() { ... }
}
The version property will be mapped to the OPTLOCK column, and the entity manager will use it to detect con-
flicting updates (preventing lost updates you might otherwise see with the last-commit-wins strategy).
The version column may be a numeric (the recommended solution) or a timestamp as per the EJB3 spec. Hi-
bernate support any kind of type provided that you define and implement the appropriate UserVersionType.
The application must not alter the version number set up by Hibernate in any way. To artificially increase the
version number, check in Hibernate Entity Manager's reference documentation LockMode.WRITE
Every non static non transient property (field or method) of an entity bean is considered persistent, unless you
annotate it as @Transient. Not having an annotation for your property is equivalent to the appropriate @Basic
annotation. The @Basic annotation allows you to declare the fetching strategy for a property:
Hibernate 3.3.0.GA 5
Entity Beans
@Transient
String getLengthInMeter() { ... } //transient property
@Basic
int getLength() { ... } // persistent property
@Basic(fetch = FetchType.LAZY)
String getDetailedComment() { ... } // persistent property
@Temporal(TemporalType.TIME)
java.util.Date getDepartureTime() { ... } // persistent property
@Enumerated(STRING)
Starred getNote() { ... } //enum persisted as String in database
counter, a transient field, and lengthInMeter, a method annotated as @Transient, and will be ignored by the
entity manager. name, length, and firstname properties are mapped persistent and eagerly fetched (the default
for simple properties). The detailedComment property value will be lazily fetched from the database once a
lazy property of the entity is accessed for the first time. Usually you don't need to lazy simple properties (not to
be confused with lazy association fetching).
Note
To enable property level lazy fetching, your classes have to be instrumented: bytecode is added to the
original one to enable such feature, please refer to the Hibernate reference documentation. If your
classes are not instrumented, property level lazy loading is silently ignored.
The recommended alternative is to use the projection capability of EJB-QL or Criteria queries.
EJB3 support property mapping of all basic types supported by Hibernate (all basic Java types , their respective
wrappers and serializable classes). Hibernate Annotations support out of the box Enum type mapping either in-
to a ordinal column (saving the enum ordinal) or a string based column (saving the enum string representation):
the persistence representation, defaulted to ordinal, can be overriden through the @Enumerated annotation as
shown in the note property example.
In core Java APIs, the temporal precision is not defined. When dealing with temporal data you might want to
describe the expected precision in database. Temporal data can have DATE, TIME, or TIMESTAMP precision (ie the
actual date, only the time, or both). Use the @Temporal annotation to fine tune that.
@Lob indicates that the property should be persisted in a Blob or a Clob depending on the property type:
java.sql.Clob, Character[], char[] and java.lang.String will be persisted in a Clob. java.sql.Blob,
Byte[], byte[] and serializable type will be persisted in a Blob.
@Lob
public String getFullText() {
return fullText;
}
@Lob
public byte[] getFullCode() {
return fullCode;
}
If the property type implements java.io.Serializable and is not a basic type, and if the property is not annot-
Hibernate 3.3.0.GA 6
Entity Beans
The column(s) used for a property mapping can be defined using the @Column annotation. Use it to override de-
fault values (see the EJB3 specification for more information on the defaults). You can use this annotation at
the property level for properties that are:
@Entity
public class Flight implements Serializable {
...
@Column(updatable = false, name = "flight_name", nullable = false, length=50)
public String getName() { ... }
The name property is mapped to the flight_name column, which is not nullable, has a length of 50 and is not
updatable (making the property immutable).
This annotation can be applied to regular properties as well as @Id or @Version properties.
@Column(
name="columnName"; (1)
boolean unique() default false; (2)
boolean nullable() default true; (3)
boolean insertable() default true; (4)
boolean updatable() default true; (5)
String columnDefinition() default ""; (6)
String table() default ""; (7)
int length() default 255; (8)
int precision() default 0; // decimal precision (9)
int scale() default 0; // decimal scale
(1) name (optional): the column name (default to the property name)
(2) unique (optional): set a unique constraint on this column or not (default false)
(3) nullable (optional): set the column as nullable (default true).
(4) insertable (optional): whether or not the column will be part of the insert statement (default true)
(5) updatable (optional): whether or not the column will be part of the update statement (default true)
(6) columnDefinition (optional): override the sql DDL fragment for this particular column (non portable)
(7) table (optional): define the targeted table (default primary table)
(8) length (optional): column length (default 255)
Hibernate 3.3.0.GA 7
Entity Beans
It is possible to declare an embedded component inside an entity and even override its column mapping. Com-
ponent classes have to be annotated at the class level with the @Embeddable annotation. It is possible to override
the column mapping of an embedded object for a particular entity using the @Embedded and
@AttributeOverride annotation in the associated property:
@Entity
public class Person implements Serializable {
@Embedded
@AttributeOverrides( {
@AttributeOverride(name="iso2", column = @Column(name="bornIso2") ),
@AttributeOverride(name="name", column = @Column(name="bornCountryName") )
} )
Country bornIn;
...
}
@Embeddable
public class Address implements Serializable {
String city;
Country nationality; //no overriding here
}
@Embeddable
public class Country implements Serializable {
private String iso2;
@Column(name="countryName") private String name;
A embeddable object inherit the access type of its owning entity (note that you can override that using the Hi-
bernate specific @AccessType annotations (see Hibernate Annotation Extensions).
The Person entity bean has two component properties, homeAddress and bornIn. homeAddress property has not
been annotated, but Hibernate will guess that it is a persistent component by looking for the @Embeddable an-
notation in the Address class. We also override the mapping of a column name (to bornCountryName) with the
@Embedded and @AttributeOverride annotations for each mapped attribute of Country. As you can see, Coun-
try is also a nested component of Address, again using auto-detection by Hibernate and EJB3 defaults. Over-
riding columns of embedded objects of embedded objects is currently not supported in the EJB3 spec, however,
Hibernate Annotations supports it through dotted expressions.
@Embedded
@AttributeOverrides( {
@AttributeOverride(name="city", column = @Column(name="fld_city") )
@AttributeOverride(name="nationality.iso2", column = @Column(name="nat_Iso2") ),
@AttributeOverride(name="nationality.name", column = @Column(name="nat_CountryName") )
//nationality columns in homeAddress are overridden
Hibernate 3.3.0.GA 8
Entity Beans
} )
Address homeAddress;
Hibernate Annotations supports one more feature that is not explicitly supported by the EJB3 specification.
You can annotate a embedded object with the @MappedSuperclass annotation to make the superclass properties
persistent (see @MappedSuperclass for more informations).
While not supported by the EJB3 specification, Hibernate Annotations allows you to use association annota-
tions in an embeddable object (ie @*ToOne nor @*ToMany). To override the association columns you can use
@AssociationOverride.
If you want to have the same embeddable object type twice in the same entity, the column name defaulting will
not work: at least one of the columns will have to be explicit. Hibernate goes beyond the EJB3 spec and allows
you to enhance the defaulting mechanism through the NamingStrategy. DefaultComponentSafeNaming-
Strategy is a small improvement over the default EJB3NamingStrategy that allows embedded objects to be de-
faulted even if used twice in the same entity.
• Otherwise, if the type of the property is Serializable, it is mapped as @Basic in a column holding the object
in its serialized version
• Otherwise, if the type of the property is java.sql.Clob or java.sql.Blob, it is mapped as @Lob with the ap-
propriate LobType
The @Id annotation lets you define which property is the identifier of your entity bean. This property can be set
by the application itself or be generated by Hibernate (preferred). You can define the identifier generation
strategy thanks to the @GeneratedValue annotation:
• SEQUENCE - sequence
Hibernate provides more id generators than the basic EJB3 ones. Check Hibernate Annotation Extensions for
more informations.
The following example shows a sequence generator using the SEQ_STORE configuration (see below)
Hibernate 3.3.0.GA 9
Entity Beans
@Id @GeneratedValue(strategy=GenerationType.IDENTITY)
public Long getId() { ... }
The AUTO generator is the preferred type for portable applications (across several DB vendors). The identifier
generation configuration can be shared for several @Id mappings with the generator attribute. There are several
configurations available through @SequenceGenerator and @TableGenerator. The scope of a generator can be
the application or the class. Class-defined generators are not visible outside the class and can override applica-
tion level generators. Application level generators are defined at XML level (see Chapter Overriding metadata
through XML, Overriding metadata through XML):
<table-generator name="EMP_GEN"
table="GENERATOR_TABLE"
pk-column-name="key"
value-column-name="hi"
pk-column-value="EMP"
allocation-size="20"/>
@javax.persistence.TableGenerator(
name="EMP_GEN",
table="GENERATOR_TABLE",
pkColumnName = "key",
valueColumnName = "hi"
pkColumnValue="EMP",
allocationSize=20
)
<sequence-generator name="SEQ_GEN"
sequence-name="my_sequence"
allocation-size="20"/>
@javax.persistence.SequenceGenerator(
name="SEQ_GEN",
sequenceName="my_sequence",
allocationSize=20
)
If JPA XML (like META-INF/orm.xml) is used to define thegenerators, EMP_GEN and SEQ_GEN are application
level generators. EMP_GEN defines a table based id generator using the hilo algorithm with a max_lo of 20. The
hi value is kept in a table "GENERATOR_TABLE". The information is kept in a row where pkColumnName "key" is
equals to pkColumnValue "EMP" and column valueColumnName "hi" contains the the next high value used.
SEQ_GEN defines a sequence generator using a sequence named my_sequence. The allocation size used for this
sequence based hilo algorithm is 20. Note that this version of Hibernate Annotations does not handle initial-
Value in the sequence generator. The default allocation size is 50, so if you want to use a sequence and pickup
the value each time, you must set the allocation size to 1.
Note
Package level definition is no longer supported by the EJB 3.0 specification. However, you can use the
@GenericGenerator at the package level (see Section 2.4.Identifier, “Identifier”).
The next example shows the definition of a sequence generator in a class scope:
Hibernate 3.3.0.GA 10
Entity Beans
@Entity
@javax.persistence.SequenceGenerator(
name="SEQ_STORE",
sequenceName="my_sequence"
)
public class Store implements Serializable {
private Long id;
This class will use a sequence named my_sequence and the SEQ_STORE generator is not visible in other
classes. Note that you can check the Hibernate Annotations tests in the org.hibernate.test.metadata.id package
for more examples.
• annotate the component property as @Id and make the component class @Embeddable
• annotate the class as @IdClass and annotate each property of the entity involved in the primary key with
@Id
While quite common to the EJB2 developer, @IdClass is likely new for Hibernate users. The composite
primary key class corresponds to multiple fields or properties of the entity class, and the names of primary key
fields or properties in the primary key class and those of the entity class must match and their types must be the
same. Let's look at an example:
@Entity
@IdClass(FootballerPk.class)
public class Footballer {
//part of the id key
@Id public String getFirstname() {
return firstname;
}
Hibernate 3.3.0.GA 11
Entity Beans
@Embeddable
public class FootballerPk implements Serializable {
//same name and type as in Footballer
public String getFirstname() {
return firstname;
}
As you may have seen, @IdClass points to the corresponding primary key class.
While not supported by the EJB3 specification, Hibernate allows you to define associations inside a composite
identifier. Simply use the regular annotations for that
@Entity
@AssociationOverride( name="id.channel", joinColumns = @JoinColumn(name="chan_id") )
public class TvMagazin {
@EmbeddedId public TvMagazinPk id;
@Temporal(TemporalType.TIME) Date time;
}
@Embeddable
public class TvMagazinPk implements Serializable {
@ManyToOne
public Channel channel;
public String name;
@ManyToOne
public Presenter presenter;
}
• Single Table per Class Hierarchy Strategy: the <subclass> element in Hibernate
The chosen strategy is declared at the class level of the top level entity in the hierarchy using the @Inheritance
annotation.
Note
Annotating interfaces is currently not supported.
Hibernate 3.3.0.GA 12
Entity Beans
This strategy has many drawbacks (esp. with polymorphic queries and associations) explained in the EJB3
spec, the Hibernate reference documentation, Hibernate in Action, and many other places. Hibernate work
around most of them implementing this strategy using SQL UNION queries. It is commonly used for the top level
of an inheritance hierarchy:
@Entity
@Inheritance(strategy = InheritanceType.TABLE_PER_CLASS)
public class Flight implements Serializable {
This strategy support one to many associations provided that they are bidirectional. This strategy does not sup-
port the IDENTITY generator strategy: the id has to be shared across several tables. Consequently, when using
this strategy, you should not use AUTO nor IDENTITY.
All properties of all super- and subclasses are mapped into the same table, instances are distinguished by a spe-
cial discriminator column:
@Entity
@Inheritance(strategy=InheritanceType.SINGLE_TABLE)
@DiscriminatorColumn(
name="planetype",
discriminatorType=DiscriminatorType.STRING
)
@DiscriminatorValue("Plane")
public class Plane { ... }
@Entity
@DiscriminatorValue("A320")
public class A320 extends Plane { ... }
Plane is the superclass, it defines the inheritance strategy InheritanceType.SINGLE_TABLE. It also defines the
discriminator column through the @DiscriminatorColumn annotation, a discriminator column can also define
the discriminator type. Finally, the @DiscriminatorValue annotation defines the value used to differentiate a
class in the hierarchy. All of these attributes have sensible default values. The default name of the discriminator
column is DTYPE. The default discriminator value is the entity name (as defined in @Entity.name) for Discrim-
inatorType.STRING. A320 is a subclass; you only have to define discriminator value if you don't want to use
the default value. The strategy and the discriminator type are implicit.
@Inheritance and @DiscriminatorColumn should only be defined at the top of the entity hierarchy.
Joined subclasses
The @PrimaryKeyJoinColumn and @PrimaryKeyJoinColumns annotations define the primary key(s) of the
joined subclass table:
@Entity
@Inheritance(strategy=InheritanceType.JOINED)
public class Boat implements Serializable { ... }
@Entity
public class Ferry extends Boat { ... }
@Entity
@PrimaryKeyJoinColumn(name="BOAT_ID")
Hibernate 3.3.0.GA 13
Entity Beans
All of the above entities use the JOINED strategy, the Ferry table is joined with the Boat table using the same
primary key names. The AmericaCupClass table is joined with Boat using the join condition Boat.id = Amer-
icaCupClass.BOAT_ID.
This is sometimes useful to share common properties through a technical or a business superclass without in-
cluding it as a regular mapped entity (ie no specific table for this entity). For that purpose you can map them as
@MappedSuperclass.
@MappedSuperclass
public class BaseEntity {
@Basic
@Temporal(TemporalType.TIMESTAMP)
public Date getLastUpdate() { ... }
public String getLastUpdater() { ... }
...
}
In database, this hierarchy will be represented as an Order table having the id, lastUpdate and lastUpdater
columns. The embedded superclass property mappings are copied into their entity subclasses. Remember that
the embeddable superclass is not the root of the hierarchy though.
Note
Properties from superclasses not mapped as @MappedSuperclass are ignored.
Note
The access type (field or methods), is inherited from the root entity, unless you use the Hibernate an-
notation @AccessType
Note
The same notion can be applied to @Embeddable objects to persist properties from their superclasses.
You also need to use @MappedSuperclass to do that (this should not be considered as a standard EJB3
feature though)
Note
It is allowed to mark a class as @MappedSuperclass in the middle of the mapped inheritance hierarchy.
Note
Any class in the hierarchy non annotated with @MappedSuperclass nor @Entity will be ignored.
You can override columns defined in entity superclasses at the root entity level using the @AttributeOverride
annotation.
Hibernate 3.3.0.GA 14
Entity Beans
@MappedSuperclass
public class FlyingObject implements Serializable {
@Transient
public int getMetricAltitude() {
return metricAltitude;
}
@ManyToOne
public PropulsionType getPropulsion() {
return metricAltitude;
}
...
}
@Entity
@AttributeOverride( name="altitude", column = @Column(name="fld_altitude") )
@AssociationOverride( name="propulsion", joinColumns = @JoinColumn(name="fld_propulsion_fk") )
public class Plane extends FlyingObject {
...
}
The altitude property will be persisted in an fld_altitude column of table Plane and the propulsion associ-
ation will be materialized in a fld_propulsion_fk foreign key column.
One-to-one
You can associate entity beans through a one-to-one relationship using @OneToOne. There are three cases for
one-to-one associations: either the associated entities share the same primary keys values, a foreign key is held
by one of the entities (note that this FK column in the database should be constrained unique to simulate one-
to-one multiplicity), or a association table is used to store the link between the 2 entities (a unique constraint
has to be defined on each fk to ensure the one to one multiplicity)
@Entity
public class Body {
@Id
public Long getId() { return id; }
@OneToOne(cascade = CascadeType.ALL)
@PrimaryKeyJoinColumn
public Heart getHeart() {
return heart;
}
...
}
@Entity
public class Heart {
@Id
Hibernate 3.3.0.GA 15
Entity Beans
In the following example, the associated entities are linked through a foreign key column:
@Entity
public class Customer implements Serializable {
@OneToOne(cascade = CascadeType.ALL)
@JoinColumn(name="passport_fk")
public Passport getPassport() {
...
}
@Entity
public class Passport implements Serializable {
@OneToOne(mappedBy = "passport")
public Customer getOwner() {
...
}
A Customer is linked to a Passport, with a foreign key column named passport_fk in the Customer table. The
join column is declared with the @JoinColumn annotation which looks like the @Column annotation. It has one
more parameters named referencedColumnName. This parameter declares the column in the targeted entity that
will be used to the join. Note that when using referencedColumnName to a non primary key column, the associated
class has to be Serializable. Also note that the referencedColumnName to a non primary key column has to be
mapped to a property having a single column (other cases might not work).
The association may be bidirectional. In a bidirectional relationship, one of the sides (and only one) has to be
the owner: the owner is responsible for the association column(s) update. To declare a side as not responsible
for the relationship, the attribute mappedBy is used. mappedBy refers to the property name of the association on
the owner side. In our case, this is passport. As you can see, you don't have to (must not) declare the join
column since it has already been declared on the owners side.
If no @JoinColumn is declared on the owner side, the defaults apply. A join column(s) will be created in the
owner table and its name will be the concatenation of the name of the relationship in the owner side, _
(underscore), and the name of the primary key column(s) in the owned side. In this example passport_id be-
cause the property name is passport and the column id of Passport is id.
@Entity
public class Customer implements Serializable {
@OneToOne(cascade = CascadeType.ALL)
@JoinTable(name = "CustomerPassports"
joinColumns = @JoinColumn(name="customer_fk"),
inverseJoinColumns = @JoinColumns(name="passport_fk")
)
public Passport getPassport() {
...
}
@Entity
public class Passport implements Serializable {
@OneToOne(mappedBy = "passport")
public Customer getOwner() {
...
Hibernate 3.3.0.GA 16
Entity Beans
A Customer is linked to a Passport through a association table named CustomerPassports ; this association ta-
ble has a foreign key column named passport_fk pointing to the Passport table (materialized by the inverse-
JoinColumn, and a foreign key column named customer_fk pointing to the Customer table materialized by the
joinColumns attribute.
You must declare the join table name and the join columns explicitly in such a mapping.
Many-to-one
Many-to-one associations are declared at the property level with the annotation @ManyToOne:
@Entity()
public class Flight implements Serializable {
@ManyToOne( cascade = {CascadeType.PERSIST, CascadeType.MERGE} )
@JoinColumn(name="COMP_ID")
public Company getCompany() {
return company;
}
...
}
The @JoinColumn attribute is optional, the default value(s) is like in one to one, the concatenation of the name
of the relationship in the owner side, _ (underscore), and the name of the primary key column in the owned
side. In this example company_id because the property name is company and the column id of Company is id.
@ManyToOne has a parameter named targetEntity which describes the target entity name. You usually don't
need this parameter since the default value (the type of the property that stores the association) is good in al-
most all cases. However this is useful when you want to use interfaces as the return type instead of the regular
entity.
@Entity()
public class Flight implements Serializable {
@ManyToOne( cascade = {CascadeType.PERSIST, CascadeType.MERGE}, targetEntity=CompanyImpl.class )
@JoinColumn(name="COMP_ID")
public Company getCompany() {
return company;
}
...
}
You can alse map a many to one association through an association table. This association table described by
the @JoinTable annotation will contains a foreign key referencing back the entity table (through
@JoinTable.joinColumns) and a a foreign key referencing the target entity table (through
@JoinTable.inverseJoinColumns).
@Entity()
public class Flight implements Serializable {
@ManyToOne( cascade = {CascadeType.PERSIST, CascadeType.MERGE} )
@JoinTable(name="Flight_Company",
joinColumns = @JoinColumn(name="FLIGHT_ID"),
inverseJoinColumns = @JoinColumns(name="COMP_ID")
Hibernate 3.3.0.GA 17
Entity Beans
)
public Company getCompany() {
return company;
}
...
}
Collections
Overview
You can map Collection, List (ie ordered lists, not indexed lists), Map and Set. The EJB3 specification de-
scribes how to map an ordered list (ie a list ordered at load time) using @javax.persistence.OrderBy annota-
tion: this annotation takes into parameter a list of comma separated (target entity) properties to order the collec-
tion by (eg firstname asc, age desc), if the string is empty, the collection will be ordered by id. For true in-
dexed collections, please refer to the Hibernate Annotation Extensions. EJB3 allows you to map Maps using as
a key one of the target entity property using @MapKey(name="myProperty") (myProperty is a property name in
the target entity). When using @MapKey (without property name), the target entity primary key is used. The map
key uses the same column as the property pointed out: there is no additional column defined to hold the map
key, and it does make sense since the map key actually represent a target property. Be aware that once loaded,
the key is no longer kept in sync with the property, in other words, if you change the property value, the key
will not change automatically in your Java model (for true map support please refers to Hibernate Annotation
Extensions). Many people confuse <map> capabilities and @MapKey ones. These are two different features.
@MapKey still has some limitations, please check the forum or the JIRA tracking system for more informations.
Hibernate 3.3.0.GA 18
Entity Beans
@org.hibernate.annotations.MapK
ey/MapKeyManyToMany for true
map support, OR
@javax.persistence.MapKey
Collection of primitive, core type or embedded objects is not supported by the EJB3 specification. Hibernate
Annotations allows them however (see Hibernate Annotation Extensions).
@ManyToOne
public City getCity() {
return city;
}
...
}
@Entity
public class Software {
@OneToMany(mappedBy="software")
@MapKey(name="codeName")
public Map<String, Version> getVersions() {
return versions;
}
...
}
@Entity
@Table(name="tbl_version")
public class Version {
public String getCodeName() {...}
@ManyToOne
public Software getSoftware() { ... }
...
}
So City has a collection of Streets that are ordered by streetName (of Street) when the collection is loaded.
Software has a map of Versions which key is the Version codeName.
Unless the collection is a generic, you will have to define targetEntity. This is a annotation attribute that take
the target entity class as a value.
One-to-many
One-to-many associations are declared at the property level with the annotation @OneToMany. One to many asso-
Hibernate 3.3.0.GA 19
Entity Beans
Bidirectional
Since many to one are (almost) always the owner side of a bidirectional relationship in the EJB3 spec, the one
to many association is annotated by @OneToMany( mappedBy=... )
@Entity
public class Troop {
@OneToMany(mappedBy="troop")
public Set<Soldier> getSoldiers() {
...
}
@Entity
public class Soldier {
@ManyToOne
@JoinColumn(name="troop_fk")
public Troop getTroop() {
...
}
Troop has a bidirectional one to many relationship with Soldier through the troop property. You don't have to
(must not) define any physical mapping in the mappedBy side.
To map a bidirectional one to many, with the one-to-many side as the owning side, you have to remove the
mappedBy element and set the many to one @JoinColumn as insertable and updatable to false. This solution is
obviously not optimized and will produce some additional UPDATE statements.
@Entity
public class Troop {
@OneToMany
@JoinColumn(name="troop_fk") //we need to duplicate the physical information
public Set<Soldier> getSoldiers() {
...
}
@Entity
public class Soldier {
@ManyToOne
@JoinColumn(name="troop_fk", insertable=false, updatable=false)
public Troop getTroop() {
...
}
Unidirectional
A unidirectional one to many using a foreign key column in the owned entity is not that common and not really
recommended. We strongly advise you to use a join table for this kind of association (as explained in the next
section). This kind of association is described through a @JoinColumn
@Entity
public class Customer implements Serializable {
@OneToMany(cascade=CascadeType.ALL, fetch=FetchType.EAGER)
@JoinColumn(name="CUST_ID")
public Set<Ticket> getTickets() {
...
}
@Entity
public class Ticket implements Serializable {
... //no bidir
}
Hibernate 3.3.0.GA 20
Entity Beans
Customer describes a unidirectional relationship with Ticket using the join column CUST_ID.
A unidirectional one to many with join table is much preferred. This association is described through an
@JoinTable.
@Entity
public class Trainer {
@OneToMany
@JoinTable(
name="TrainedMonkeys",
joinColumns = { @JoinColumn( name="trainer_id") },
inverseJoinColumns = @JoinColumn( name="monkey_id")
)
public Set<Monkey> getTrainedMonkeys() {
...
}
@Entity
public class Monkey {
... //no bidir
}
Trainer describes a unidirectional relationship with Monkey using the join table TrainedMonkeys, with a for-
eign key trainer_id to Trainer (joinColumns) and a foreign key monkey_id to Monkey
(inversejoinColumns).
Defaults
Without describing any physical mapping, a unidirectional one to many with join table is used. The table name
is the concatenation of the owner table name, _, and the other side table name. The foreign key name(s) referen-
cing the owner table is the concatenation of the owner table, _, and the owner primary key column(s) name.
The foreign key name(s) referencing the other side is the concatenation of the owner property name, _, and the
other side primary key column(s) name. A unique constraint is added to the foreign key referencing the other
side table to reflect the one to many.
@Entity
public class Trainer {
@OneToMany
public Set<Tiger> getTrainedTigers() {
...
}
@Entity
public class Tiger {
... //no bidir
}
Trainer describes a unidirectional relationship with Tiger using the join table Trainer_Tiger, with a foreign
key trainer_id to Trainer (table name, _, trainer id) and a foreign key trainedTigers_id to Monkey (property
name, _, Tiger primary column).
Many-to-many
Definition
Hibernate 3.3.0.GA 21
Entity Beans
A many-to-many association is defined logically using the @ManyToMany annotation. You also have to describe
the association table and the join conditions using the @JoinTable annotation. If the association is bidirectional,
one side has to be the owner and one side has to be the inverse end (ie. it will be ignored when updating the re-
lationship values in the association table):
@Entity
public class Employer implements Serializable {
@ManyToMany(
targetEntity=org.hibernate.test.metadata.manytomany.Employee.class,
cascade={CascadeType.PERSIST, CascadeType.MERGE}
)
@JoinTable(
name="EMPLOYER_EMPLOYEE",
joinColumns={@JoinColumn(name="EMPER_ID")},
inverseJoinColumns={@JoinColumn(name="EMPEE_ID")}
)
public Collection getEmployees() {
return employees;
}
...
}
@Entity
public class Employee implements Serializable {
@ManyToMany(
cascade={CascadeType.PERSIST, CascadeType.MERGE},
mappedBy="employees"
targetEntity=Employer.class
)
public Collection getEmployers() {
return employers;
}
}
We've already shown the many declarations and the detailed attributes for associations. We'll go deeper in the
@JoinTable description, it defines a name, an array of join columns (an array in annotation is defined using { A,
B, C }), and an array of inverse join columns. The latter ones are the columns of the association table which
refer to the Employee primary key (the "other side").
As seen previously, the other side don't have to (must not) describe the physical mapping: a simple mappedBy
argument containing the owner side property name bind the two.
Default values
As any other annotations, most values are guessed in a many to many relationship. Without describing any
physical mapping in a unidirectional many to many the following rules applied. The table name is the concaten-
ation of the owner table name, _ and the other side table name. The foreign key name(s) referencing the owner
table is the concatenation of the owner table name, _ and the owner primary key column(s). The foreign key
name(s) referencing the other side is the concatenation of the owner property name, _, and the other side
primary key column(s). These are the same rules used for a unidirectional one to many relationship.
@Entity
public class Store {
@ManyToMany(cascade = CascadeType.PERSIST)
public Set<City> getImplantedIn() {
...
}
}
Hibernate 3.3.0.GA 22
Entity Beans
@Entity
public class City {
... //no bidirectional relationship
}
A Store_City is used as the join table. The Store_id column is a foreign key to the Store table. The im-
plantedIn_id column is a foreign key to the City table.
Without describing any physical mapping in a bidirectional many to many the following rules applied. The ta-
ble name is the concatenation of the owner table name, _ and the other side table name. The foreign key
name(s) referencing the owner table is the concatenation of the other side property name, _, and the owner
primary key column(s). The foreign key name(s) referencing the other side is the concatenation of the owner
property name, _, and the other side primary key column(s). These are the same rules used for a unidirectional
one to many relationship.
@Entity
public class Store {
@ManyToMany(cascade = {CascadeType.PERSIST, CascadeType.MERGE})
public Set<Customer> getCustomers() {
...
}
}
@Entity
public class Customer {
@ManyToMany(mappedBy="customers")
public Set<Store> getStores() {
...
}
}
A Store_Customer is used as the join table. The stores_id column is a foreign key to the Store table. The
customers_id column is a foreign key to the Customer table.
You probably have noticed the cascade attribute taking an array of CascadeType as a value. The cascade
concept in EJB3 is very is similar to the transitive persistence and cascading of operations in Hibernate, but
with slightly different semantics and cascading types:
• CascadeType.PERSIST: cascades the persist (create) operation to associated entities persist() is called or if
the entity is managed
• CascadeType.MERGE: cascades the merge operation to associated entities if merge() is called or if the en-
tity is managed
Note
Hibernate 3.3.0.GA 23
Entity Beans
CascadeType.ALL also covers Hibernate specific operations like save-update, lock etc... Check Cas-
cade for more information
Please refer to the chapter 6.3 of the EJB3 specification for more information on cascading and create/merge
semantics.
Association fetching
You have the ability to either eagerly or lazily fetch associated entities. The fetch parameter can be set to
FetchType.LAZY or FetchType.EAGER. EAGER will try to use an outer join select to retrieve the associated ob-
ject, while LAZY will only trigger a select when the associated object is accessed for the first time. @OneToMany
and @ManyToMany associations are defaulted to LAZY and @OneToOne and @ManyToOne are defaulted to EAGER. For
more information about static fetching, check Section 2.4.5.Lazy options and fetching modes, “Lazy options
and fetching modes”.
The recommanded approach is to use LAZY onn all static fetching definitions and override this choice dynamic-
ally through JPA-QL. JPA-QL has a fetch keyword that allows you to override laziness when doing a particu-
lar query. This is very useful to improve performance and is decided on a use case to use case basis.
Composite primary keys use a embedded class as the primary key representation, so you'd use the @Id and
@Embeddable annotations. Alternatively, you can use the @EmbeddedId annotation. Note that the dependent class
has to be serializable and implements equals()/hashCode(). You can also use @IdClass as described in Map-
ping identifier properties.
@Entity
public class RegionalArticle implements Serializable {
@Id
public RegionalArticlePk getPk() { ... }
}
@Embeddable
public class RegionalArticlePk implements Serializable { ... }
or alternatively
@Entity
public class RegionalArticle implements Serializable {
@EmbeddedId
public RegionalArticlePk getPk() { ... }
}
@Embeddable inherit the access type of its owning entity unless the Hibernate specific annotation @AccessType
is used. Composite foreign keys (if not using the default sensitive values) are defined on associations using the
@JoinColumns element, which is basically an array of @JoinColumn. It is considered a good practice to express
referencedColumnNames explicitly. Otherwise, Hibernate will suppose that you use the same order of columns
as in the primary key declaration.
Hibernate 3.3.0.GA 24
Entity Beans
@Entity
public class Parent implements Serializable {
@Id
public ParentPk id;
public int age;
@OneToMany(cascade=CascadeType.ALL)
@JoinColumns ({
@JoinColumn(name="parentCivility", referencedColumnName = "isMale"),
@JoinColumn(name="parentLastName", referencedColumnName = "lastName"),
@JoinColumn(name="parentFirstName", referencedColumnName = "firstName")
})
public Set<Child> children; //unidirectional
...
}
@Entity
public class Child implements Serializable {
@Id @GeneratedValue
public Integer id;
@ManyToOne
@JoinColumns ({
@JoinColumn(name="parentCivility", referencedColumnName = "isMale"),
@JoinColumn(name="parentLastName", referencedColumnName = "lastName"),
@JoinColumn(name="parentFirstName", referencedColumnName = "firstName")
})
public Parent parent; //unidirectional
}
@Embeddable
public class ParentPk implements Serializable {
String firstName;
String lastName;
...
}
You can map a single entity bean to several tables using the @SecondaryTable or @SecondaryTables class level
annotations. To express that a column is in a particular table, use the table parameter of @Column or
@JoinColumn.
@Entity
@Table(name="MainCat")
@SecondaryTables({
@SecondaryTable(name="Cat1", pkJoinColumns={
@PrimaryKeyJoinColumn(name="cat_id", referencedColumnName="id")
),
@SecondaryTable(name="Cat2", uniqueConstraints={@UniqueConstraint(columnNames={"storyPart2"})})
})
public class Cat implements Serializable {
Hibernate 3.3.0.GA 25
Entity Beans
@Id @GeneratedValue
public Integer getId() {
return id;
}
@Column(table="Cat1")
public String getStoryPart1() {
return storyPart1;
}
@Column(table="Cat2")
public String getStoryPart2() {
return storyPart2;
}
In this example, name will be in MainCat. storyPart1 will be in Cat1 and storyPart2 will be in Cat2. Cat1
will be joined to MainCat using the cat_id as a foreign key, and Cat2 using id (ie the same column name, the
MainCat id column has). Plus a unique constraint on storyPart2 has been set.
Check out the JBoss EJB 3 tutorial or the Hibernate Annotations unit test suite for more examples.
You can map EJBQL/HQL queries using annotations. @NamedQuery and @NamedQueries can be defined at the
class level or in a JPA XML file. However their definitions are global to the session factory/entity manager
factory scope. A named query is defined by its name and the actual query string.
<entity-mappings>
<named-query name="plane.getAll">
<query>select p from Plane p</query>
</named-query>
...
</entity-mappings>
...
@Entity
@NamedQuery(name="night.moreRecentThan", query="select n from Night n where n.date >= :date")
public class Night {
...
}
You can also provide some hints to a query through an array of QueryHint through a hints attribute.
Hibernate 3.3.0.GA 26
Entity Beans
hint description
You can also map a native query (ie a plain SQL query). To achieve that, you need to describe the SQL result-
set structure using @SqlResultSetMapping (or @SqlResultSetMappings if you plan to define several resulset
mappings). Like @NamedQuery, a @SqlResultSetMapping can be defined at class level or in a JPA XML file.
However its scope is global to the application.
In the above example, the night&area named query use the joinMapping result set mapping. This mapping re-
turns 2 entities, Night and Area, each property is declared and associated to a column name, actually the
column name retrieved by the query. Let's now see an implicit declaration of the property / column.
@Entity
Hibernate 3.3.0.GA 27
Entity Beans
@SqlResultSetMapping(name="implicit", entities=@EntityResult(entityClass=org.hibernate.test.annotation
@NamedNativeQuery(name="implicitSample", query="select * from SpaceShip", resultSetMapping="implicit")
public class SpaceShip {
private String name;
private String model;
private double speed;
@Id
public String getName() {
return name;
}
@Column(name="model_txt")
public String getModel() {
return model;
}
In this example, we only describe the entity member of the result set mapping. The property / column mappings
is done using the entity mapping values. In this case the model property is bound to the model_txt column. If
the association to a related entity involve a composite primary key, a @FieldResult element should be used for
each foreign key column. The @FieldResult name is composed of the property name for the relationship, fol-
lowed by a dot ("."), followed by the name or the field or property of the primary key.
@Entity
@SqlResultSetMapping(name="compositekey",
entities=@EntityResult(entityClass=org.hibernate.test.annotations.query.SpaceShip.class,
fields = {
@FieldResult(name="name", column = "name"),
@FieldResult(name="model", column = "model"),
@FieldResult(name="speed", column = "speed"),
@FieldResult(name="captain.firstname", column = "firstn"),
@FieldResult(name="captain.lastname", column = "lastn"),
@FieldResult(name="dimensions.length", column = "length"),
@FieldResult(name="dimensions.width", column = "width")
}),
columns = { @ColumnResult(name = "surface"),
@ColumnResult(name = "volume") } )
@NamedNativeQuery(name="compositekey",
query="select name, model, speed, lname as lastn, fname as firstn, length, width, length * width a
resultSetMapping="compositekey")
} )
public class SpaceShip {
private String name;
private String model;
private double speed;
private Captain captain;
private Dimensions dimensions;
@Id
public String getName() {
Hibernate 3.3.0.GA 28
Entity Beans
return name;
}
@ManyToOne(fetch= FetchType.LAZY)
@JoinColumns( {
@JoinColumn(name="fname", referencedColumnName = "firstname"),
@JoinColumn(name="lname", referencedColumnName = "lastname")
} )
public Captain getCaptain() {
return captain;
}
@Entity
@IdClass(Identity.class)
public class Captain implements Serializable {
private String firstname;
private String lastname;
@Id
public String getFirstname() {
return firstname;
}
@Id
public String getLastname() {
return lastname;
}
Hibernate 3.3.0.GA 29
Entity Beans
Note
If you look at the dimension property, you'll see that Hibernate supports the dotted notation for embed-
ded objects (you can even have nested embedded objects). EJB3 implementations do not have to sup-
port this feature, we do :-)
If you retrieve a single entity and if you use the default mapping, you can use the resultClass attribute instead
of resultSetMapping:
In some of your native queries, you'll have to return scalar values, for example when building report queries.
You can map them in the @SqlResultsetMapping through @ColumnResult. You actually can even mix, entities
and scalar returns in the same native query (this is probably not that common though).
@SqlResultSetMapping(name="scalar", columns=@ColumnResult(name="dimension"))
@NamedNativeQuery(name="scalar", query="select length*width as dimension from SpaceShip", resultSetMap
An other query hint specific to native queries has been introduced: org.hibernate.callable which can be true
or false depending on whether the query is a stored procedure or not.
To empower the EJB3 capabilities, hibernate provides specific annotations that match hibernate features. The
org.hibernate.annotations package contains all these annotations extensions.
2.4.1. Entity
You can fine tune some of the actions done by Hibernate on entities beyond what the EJB3 spec offers.
@org.hibernate.annotations.Entity adds additional metadata that may be needed beyond what is defined in
the standard @Entity
• selectBeforeUpdate: Specifies that Hibernate should never perform an SQL UPDATE unless it is certain
that an object is actually modified.
Hibernate 3.3.0.GA 30
Entity Beans
Note
@javax.persistence.Entity is still mandatory, @org.hibernate.annotations.Entity is not a replacement.
@org.hibernate.annotations.BatchSize allows you to define the batch size when fetching instances of this
entity ( eg. @BatchSize(size=4) ). When loading a given entity, Hibernate will then load all the uninitialized
entities of the same type in the persistence context up to the batch size.
@org.hibernate.annotations.Proxy defines the laziness attributes of the entity. lazy (default to true) define
whether the class is lazy or not. proxyClassName is the interface used to generate the proxy (default is the class
itself).
@org.hibernate.annotations.Where defines an optional SQL WHERE clause used when instances of this
class is retrieved.
Note
@org.hibernate.annotations.Table is a complement, not a replacement to
@javax.persistence.Table. Especially, if you want to change the default name of a table, you must
use @javax.persistence.Table, not @org.hibernate.annotations.Table.
@org.hibernate.annotations.Table can also be used to define the following elements of secondary tables:
• fetch: If set to JOIN, the default, Hibernate will use an inner join to retrieve a secondary table defined by a
class or its superclasses and an outer join for a secondary table defined by a subclass. If set to select then
Hibernate will use a sequential select for a secondary table defined on a subclass, which will be issued only
if a row turns out to represent an instance of the subclass. Inner joins will still be used to retrieve a second-
ary defined by the class and its superclasses.
• inverse: If true, Hibernate will not try to insert or update the properties defined by this join. Default to
false.
• optional: If enabled (the default), Hibernate will insert a row only if the properties defined by this join are
non-null and will always use an outer join to retrieve the properties.
• foreignKey: defines the Foreign Key name of a secondary table pointing back to the primary table.
@Immutable marks an entity as immutable. The entity may not be updated or deleted by the application. This al-
lows Hibernate to make some minor performance optimizations.
@Persister lets you define your own custom persistence strategy. You may, for example, specify your own
Hibernate 3.3.0.GA 31
Entity Beans
@Entity
@BatchSize(size=5)
@org.hibernate.annotations.Entity(
selectBeforeUpdate = true,
dynamicInsert = true, dynamicUpdate = true,
optimisticLock = OptimisticLockType.ALL,
polymorphism = PolymorphismType.EXPLICIT)
@Where(clause="1=1")
@org.hibernate.annotations.Table(name="Forest", indexes = { @Index(name="idx", columnNames = { "name",
@Persister(impl=MyEntityPersister.class)
public class Forest { ... }
@Entity
@Inheritance(
strategy=InheritanceType.JOINED
)
public class Vegetable { ... }
@Entity
@OnDelete(action=OnDeleteAction.CASCADE)
public class Carrot extends Vegetable { ... }
2.4.Identifier. Identifier
@Id @GeneratedValue(generator="system-uuid")
@GenericGenerator(name="system-uuid", strategy = "uuid")
public String getId() {
@Id @GeneratedValue(generator="hibseq")
@GenericGenerator(name="hibseq", strategy = "seqhilo",
parameters = {
@Parameter(name="max_lo", value = "5"),
@Parameter(name="sequence", value="heybabyhey")
}
)
public Integer getId() {
strategy is the short name of an Hibernate3 generator strategy or the fully qualified class name of an Identi-
fierGenerator implementation. You can add some parameters through the parameters attribute.
Contrary to its standard counterpart, @GenericGenerator can be used in package level annotations, making it
an application level generator (just like if it were in a JPA XML file).
2.4.3. Property
Hibernate 3.3.0.GA 32
Entity Beans
Access type
The access type is guessed from the position of @Id or @EmbeddedId in the entity hierarchy. Sub-entities, em-
bedded objects and mapped superclass inherit the access type from the root entity.
• fine tune the access type at the class level or at the property level
An @AccessType annotation has been introduced to support this behavior. You can define the access type on
• an entity
• a superclass
• an embeddable object
• a property
The access type is overriden for the annotated element, if overriden on a class, all the properties of the given
class inherit the access type. For root entities, the access type is considered to be the default one for the whole
hierarchy (overridable at class or property level).
If the access type is marked as "property", the getters are scanned for annotations, if the access type is marked
as "field", the fields are scanned for annotations. Otherwise the elements marked with @Id or @embeddedId
are scanned.
You can override an access type for a property, but the element to annotate will not be influenced: for example
an entity having access type field, can annotate a field with @AccessType("property"), the access type will
then be property for this attribute, the the annotations still have to be carried on the field.
If a superclass or an embeddable object is not annotated, the root entity access type is used (even if an access
type has been define on an intermediate superclass or embeddable object). The russian doll principle does not
apply.
@Entity
public class Person implements Serializable {
@Id @GeneratedValue //access type field
Integer id;
@Embedded
@AttributeOverrides({
@AttributeOverride(name = "iso2", column = @Column(name = "bornIso2")),
@AttributeOverride(name = "name", column = @Column(name = "bornCountryName"))
})
Country bornIn;
}
@Embeddable
@AccessType("property") //override access type for all properties in Country
public class Country implements Serializable {
private String iso2;
private String name;
Hibernate 3.3.0.GA 33
Entity Beans
@Column(name = "countryName")
public String getName() {
return name;
}
Formula
Sometimes, you want the Database to do some computation for you rather than in the JVM, you might also cre-
ate some kind of virtual column. You can use a SQL fragment (aka formula) instead of mapping a property into
a column. This kind of property is read only (its value is calculated by your formula fragment).
The SQL fragment can be as complex as you want avec even include subselects.
Type
@org.hibernate.annotations.Type overrides the default hibernate type used: this is generally not necessary
since the type is correctly inferred by Hibernate. Please refer to the Hibernate reference guide for more inform-
ations on the Hibernate types.
@TypeDefs(
{
@TypeDef(
name="caster",
typeClass = CasterStringType.class,
parameters = {
@Parameter(name="cast", value="lower")
}
)
}
)
package org.hibernate.test.annotations.entity;
...
public class Forest {
@Type(type="caster")
public String getSmallText() {
...
}
When using composite user type, you will have to express column definitions. The @Columns has been intro-
duced for that purpose.
@Type(type="org.hibernate.test.annotations.entity.MonetaryAmountUserType")
Hibernate 3.3.0.GA 34
Entity Beans
@Columns(columns = {
@Column(name="r_amount"),
@Column(name="r_currency")
})
public MonetaryAmount getAmount() {
return amount;
}
Index
You can define an index on a particular column using the @Index annotation on a one column property, the
columnNames attribute will then be ignored
@Column(secondaryTable="Cat1")
@Index(name="story1index")
public String getStoryPart1() {
return storyPart1;
}
@Parent
When inside an embeddable object, you can define one of the properties as a pointer back to the owner element.
@Entity
public class Person {
@Embeddable public Address address;
...
}
@Embeddable
public class Address {
@Parent public Person owner;
...
}
person == person.address.owner
Generated properties
Some properties are generated at insert or update time by your database. Hibernate can deal with such proper-
ties and triggers a subsequent select to read these properties.
@Entity
public class Antenna {
@Id public Integer id;
@Generated(GenerationTime.ALWAYS) @Column(insertable = false, updatable = false)
public String longitude;
Annotate your property as @Generated You have to make sure your insertability or updatability does not con-
flict with the generation strategy you have chosen. When GenerationTime.INSERT is chosen, the property
Hibernate 3.3.0.GA 35
Entity Beans
must not contains insertable columns, when GenerationTime.ALWAYS is chosen, the property must not con-
tains insertable nor updatable columns.
@Target
Sometimes, the type guessed by reflection is not the one you want Hibernate to use. This is especially true on
components when an interface is used. You can use @Target to by pass the reflection guessing mechanism
(very much like the targetEntity attribute available on associations.
@Embedded
@Target(OwnerImpl.class)
public Owner getOwner() {
return owner;
}
Optimistic lock
It is sometimes useful to avoid increasing the version number even if a given property is dirty (particularly col-
lections). You can do that by annotating the property (or collection) with @OptimisticLock(excluded=true).
More formally, specifies that updates to this property do not require acquisition of the optimistic lock.
2.4.4. Inheritance
SINGLE_TABLE is a very powerful strategy but sometimes, and especially for legacy systems, you cannot add
an additional discriminator column. For that purpose Hibernate has introduced the notion of discriminator for-
mula: @DiscriminatorFormula is a replacement of @DiscriminatorColumn and use a SQL fragment as a for-
mula for discriminator resolution (no need to have a dedicated column).
@Entity
@DiscriminatorFormula("case when forest_type is null then 0 else forest_type end")
public class Forest { ... }
By default, when querying the top entities, Hibernate does not put a restriction clause on the discriminator
column. This can be inconvenient if this column contains values not mapped in your hierarchy (through
@DiscriminatorValue). To work around that you can use @ForceDiscriminator (at the class level, next to
@DiscriminatorColumn). Hibernate will then list the available values when loading the entities.
You can define the foreign key name generated by Hibernate for subclass tables in the JOINED inheritance
strategy.
@Entity
@Inheritance(strategy = InheritanceType.JOINED)
public abstract class File { ... }
@Entity
@ForeignKey(name = "FK_DOCU_FILE")
public class Document extends File {
The foreign key from the Document table to the File table will be named FK_DOCU_FILE.
Hibernate 3.3.0.GA 36
Entity Beans
By default, when Hibernate cannot resolve the association because the expected associated element is not in
database (wrong id on the association column), an exception is raised by Hibernate. This might be inconvenient
for lecacy and badly maintained schemas. You can ask Hibernate to ignore such elements instead of raising an
exception using the @NotFound annotation. This annotation can be used on a @OneToOne (with FK), @ManyToOne,
@OneToMany or @ManyToMany association.
@Entity
public class Child {
...
@ManyToOne
@NotFound(action=NotFoundAction.IGNORE)
public Parent getParent() { ... }
...
}
Sometimes you want to delegate to your database the deletion of cascade when a given entity is deleted.
@Entity
public class Child {
...
@ManyToOne
@OnDelete(action=OnDeleteAction.CASCADE)
public Parent getParent() { ... }
...
}
In this case Hibernate generates a cascade delete constraint at the database level.
Foreign key constraints, while generated by Hibernate, have a fairly unreadable name. You can override the
constraint name by use @ForeignKey.
@Entity
public class Child {
...
@ManyToOne
@ForeignKey(name="FK_PARENT")
public Parent getParent() { ... }
...
}
alter table Child add constraint FK_PARENT foreign key (parent_id) references Parent
EJB3 comes with the fetch option to define lazy loading and fetching modes, however Hibernate has a much
more option set in this area. To fine tune the lazy loading and fetching strategies, some additional annotations
have been introduced:
• @LazyToOne: defines the lazyness option on @ManyToOne and @OneToOne associations. LazyToOneOption can
be PROXY (ie use a proxy based lazy loading), NO_PROXY (use a bytecode enhancement based lazy loading -
note that build time bytecode processing is necessary) and FALSE (association not lazy)
• @LazyCollection: defines the lazyness option on @ManyToMany and @OneToMany associations. LazyCollec-
tionOption can be TRUE (the collection is lazy and will be loaded when its state is accessed), EXTRA (the col-
lection is lazy and all operations will try to avoid the collection loading, this is especially useful for huge
collections when loading all the elements is not necessary) and FALSE (association not lazy)
• @Fetch: defines the fetching strategy used to load the association. FetchMode can be SELECT (a select is
triggered when the association needs to be loaded), SUBSELECT (only available for collections, use a subse-
Hibernate 3.3.0.GA 37
Entity Beans
lect strategy - please refers to the Hibernate Reference Documentation for more information) or JOIN (use a
SQL JOIN to load the association while loading the owner entity). JOIN overrides any lazy attribute (an as-
sociation loaded through a JOIN strategy cannot be lazy).
It is possible to set
• the where clause, using @Where (applied on the target entity) or @WhereJoinTable (applied on the associ-
ation table)
• the collection immutability using @Immutable: if set specifies that the elements of the collection never
change (a minor performance optimization in some cases)
• a custom collection persister (ie the persistence strategy used) using @Persister: the class must implement
org.hibernate.persister.collectionCollectionPersister
You can also declare a sort comparator. Use the @Sort annotation. Expressing the comparator type you want
between unsorted, natural or custom comparator. If you want to use your own comparator implementation,
you'll also have to express the implementation class using the comparator attribute. Note that you need to use
either a SortedSet or a SortedMap interface.
@OneToMany(cascade=CascadeType.ALL, fetch=FetchType.EAGER)
@JoinColumn(name="CUST_ID")
@Sort(type = SortType.COMPARATOR, comparator = TicketComparator.class)
@Where(clause="1=1")
Hibernate 3.3.0.GA 38
Entity Beans
@OnDelete(action=OnDeleteAction.CASCADE)
public SortedSet<Ticket> getTickets() {
return tickets;
}
Please refer to the previous descriptions of these annotations for more informations.
Foreign key constraints, while generated by Hibernate, have a fairly unreadable name. You can override the
constraint name by use @ForeignKey. Note that this annotation has to be placed on the owning side of the rela-
tionship, inverseName referencing to the other side constraint.
@Entity
public class Woman {
...
@ManyToMany(cascade = {CascadeType.ALL})
@ForeignKey(name = "TO_WOMAN_FK", inverseName = "TO_MAN_FK")
public Set<Man> getMens() {
return mens;
}
}
alter table Man_Woman add constraint TO_WOMAN_FK foreign key (woman_id) references Woman
alter table Man_Woman add constraint TO_MAN_FK foreign key (man_id) references Man
List
Beyond EJB3, Hibernate Annotations supports true List and Array. Map your collection the same way as usual
and add the @IndexColumn. This annotation allows you to describe the column that will hold the index. You
can also declare the index value in DB that represent the first element (aka as base index). The usual value is 0
or 1.
@OneToMany(cascade = CascadeType.ALL)
@IndexColumn(name = "drawer_position", base=1)
public List<Drawer> getDrawers() {
return drawers;
}
Note
If you forgot to set @IndexColumn, the bag semantic is applied. If you want the bag semantic without
the limitations of it, consider using @CollectionId.
Map
Hibernate Annotations also supports true Map mappings, if @javax.persistence.MapKey is not set, hibernate
will map the key element or embeddable object in its/their own columns. To overrides the default columns, you
can use @org.hibernate.annotations.MapKey if your key is a basic type (defaulted to mapkey) or an embed-
dable object, or you can use @org.hibernate.annotations.MapKeyManyToMany if your key is an entity.
@CollectionOfElements(targetElement = SizeImpl.class)
@MapKeyManyToMany(targetEntity = LuggageImpl.class)
private Map<Luggage, Size> sizePerLuggage = new HashMap<Luggage, Size>();
Hibernate 3.3.0.GA 39
Entity Beans
@Entity
public class Parent {
@OneToMany(mappedBy="parent")
@org.hibernate.annotations.MapKey(columns=@Column(name="name"))
private Map<String, Child> children;
...
}
@Entity
public class Parent {
...
@Basic
private String name;
@ManyToOne
@JoinColumn(name="parent_id", nullable=false)
private Parent parent;
...
}
But, if there is no such property on the child class, we can't think of the association as truly bidirectional (there
is information available at one end of the association that is not available at the other end). In this case, we can't
map the collection mappedBy. Instead, we could use the following mapping:
@Entity
public class Parent {
@OneToMany
@org.hibernate.annotations.MapKey(columns=@Column(name="name"))
@JoinColumn(name="parent_id", nullable=false)
private Map<String, Child> children;
...
}
@Entity
public class Parent {
...
@ManyToOne
@JoinColumn(name="parent_id", insertable=false, updatable=false, nullable=false)
private Parent parent;
...
}
Note that in this mapping, the collection-valued end of the association is responsible for updates to the foreign
key.
@Entity
Hibernate 3.3.0.GA 40
Entity Beans
@TableGenerator(name="ids_generator", table="IDS")
public class Passport {
...
@ManyToMany(cascade = CascadeType.ALL)
@JoinTable(name="PASSPORT_VISASTAMP")
@CollectionId(
columns = @Column(name="COLLECTION_ID"),
type=@Type(type="long"),
generator = "ids_generator"
)
private Collection<Stamp> visaStamp = new ArrayList();
...
}
@Entity
public class Boy {
private Integer id;
private Set<String> nickNames = new HashSet<String>();
private int[] favoriteNumbers;
private Set<Toy> favoriteToys = new HashSet<Toy>();
private Set<Character> characters = new HashSet<Character>();
@Id @GeneratedValue
public Integer getId() {
return id;
}
@CollectionOfElements
public Set<String> getNickNames() {
return nickNames;
}
@CollectionOfElements
@JoinTable(
table=@Table(name="BoyFavoriteNumbers"),
joinColumns = @JoinColumn(name="BoyId")
)
@Column(name="favoriteNumber", nullable=false)
@IndexColumn(name="nbr_index")
public int[] getFavoriteNumbers() {
return favoriteNumbers;
}
@CollectionOfElements
@AttributeOverride( name="element.serial", column=@Column(name="serial_nbr") )
public Set<Toy> getFavoriteToys() {
return favoriteToys;
}
@CollectionOfElements
public Set<Character> getCharacters() {
Hibernate 3.3.0.GA 41
Entity Beans
return characters;
}
...
}
@Embeddable
public class Toy {
public String name;
public String serial;
public Boy owner;
@Parent
public Boy getOwner() {
return owner;
}
return true;
}
On a collection of embeddable objects, the embeddable object can have a property annotated with @Parent.
This property will then point back to the entity containing the collection.
Note
Hibernate 3.3.0.GA 42
Entity Beans
Previous versions of Hibernate Annotations used the @OneToMany to mark a collection of elements. Due
to semantic inconsistencies, we've introduced the annotation @CollectionOfElements. Marking collec-
tions of elements the old way still work but is considered deprecated and is going to be unsupported in
future releases
2.4.7. Cascade
Hibernate offers more operations than the Java Persistence specification. You can use the @Cascade annotation
to cascade the following operations:
• PERSIST
• MERGE
• REMOVE
• REFRESH
• DELETE
• SAVE_UPDATE
• REPLICATE
• DELETE_ORPHAN
• LOCK
• EVICT
This is especially useful for SAVE_UPDATE (which is the operation cascaded at flush time if you use plain Hi-
bernate Annotations - Hibernate EntityManager cascade PERSIST at flush time as per the specification). DE-
LETE_ORPHAN applies only to @OneToMany associations, and indicates that the delete()/remove() operation
should be applied to any child object that is removed from the association. In other words, if a child is derefer-
enced by a persistent parent and if DELETE_ORPHAN is used, the "orphaned" child is deleted.
2.4.8. Cache
In order to optimize your database accesses, you can activate the so called second level cache of Hibernate.
This cache is configurable on a per entity and per collection basis.
@org.hibernate.annotations.Cache defines the caching strategy and region of a given second level cache.
This annotation can be applied on the root entity (not the sub entities), and on the collections.
@Entity
@Cache(usage = CacheConcurrencyStrategy.NONSTRICT_READ_WRITE)
public class Forest { ... }
Hibernate 3.3.0.GA 43
Entity Beans
@OneToMany(cascade=CascadeType.ALL, fetch=FetchType.EAGER)
@JoinColumn(name="CUST_ID")
@Cache(usage = CacheConcurrencyStrategy.NONSTRICT_READ_WRITE)
public SortedSet<Ticket> getTickets() {
return tickets;
}
@Cache(
CacheConcurrencyStrategy usage(); (1)
String region() default ""; (2)
String include() default "all"; (3)
)
(1) usage: the given cache concurrency strategy (NONE, READ_ONLY, NONSTRICT_READ_WRITE,
READ_WRITE, TRANSACTIONAL)
(2) region (optional): the cache region (default to the fqcn of the class or the fq role name of the collection)
(3) include (optional): all to include all properties, non-lazy to only include non lazy properties (default all).
2.4.9. Filters
Hibernate has the ability to apply arbitrary filters on top of your data. Those filters are applied at runtime on a
given session. First, you need to define them.
We now need to define the SQL filter clause applied to either the entity load or the collection load. @Filter is
used and placed either on the entity or the collection element
@Entity
@FilterDef(name="minLength", parameters={ @ParamDef( name="minLength", type="integer" ) } )
@Filters( {
@Filter(name="betweenLength", condition=":minLength <= length and :maxLength >= length"),
@Filter(name="minLength", condition=":minLength <= length")
} )
public class Forest { ... }
When the collection use an association table as a relational representation, you might want to apply the filter
condition to the association table itself or to the target entity table. To apply the constraint on the target entity,
use the regular @Filter annotation. However, if you wan to target the association table, use the
@FilterJoinTable annotation.
@OneToMany
@JoinTable
//filter on the target entity table
@Filter(name="betweenLength", condition=":minLength <= length and :maxLength >= length")
//filter on the association table
@FilterJoinTable(name="security", condition=":userlevel >= requredLevel")
public Set<Forest> getForests() { ... }
2.4.10. Queries
Hibernate 3.3.0.GA 44
Entity Beans
Since Hibernate has more features on named queries than the one defined in the EJB3 specification,
@org.hibernate.annotations.NamedQuery, @org.hibernate.annotations.NamedQueries,
@org.hibernate.annotations.NamedNativeQuery and @org.hibernate.annotations.NamedNativeQueries
have been introduced. They add some attributes to the standard version and can be used as a replacement:
• flushMode: define the query flush mode (Always, Auto, Commit or Never)
• callable: for native queries only, to be set to true for stored procedures
• comment: if comments are activated, the comment seen when the query is sent to the database.
• readOnly: whether or not the elements retrievent from the query are in read only mode.
Those hints can be set in a standard @javax.persistence.NamedQuery annotations through the detyped
@QueryHint. Another key advantage is the ability to set those annotations at a package level.
Hibernate gives you the avility to override every single SQL statement generated. We have seen native SQL
query usage already, but you can also override the SQL statement used to load or change the state of entities.
@Entity
@Table(name="CHAOS")
@SQLInsert( sql="INSERT INTO CHAOS(size, name, nickname, id) VALUES(?,upper(?),?,?)")
@SQLUpdate( sql="UPDATE CHAOS SET size = ?, name = upper(?), nickname = ? WHERE id = ?")
@SQLDelete( sql="DELETE CHAOS WHERE id = ?")
@SQLDeleteAll( sql="DELETE CHAOS")
@Loader(namedQuery = "chaos")
@NamedNativeQuery(name="chaos", query="select id, size, name, lower( nickname ) as nickname from CHAOS
public class Chaos {
@Id
private Long id;
private Long size;
private String name;
private String nickname;
@SQLInsert, @SQLUpdate, @SQLDelete, @SQLDeleteAll respectively override the INSERT statement, UPDATE
statement, DELETE statement, DELETE statement to remove all entities.
If you expect to call a store procedure, be sure to set the callable attribute to true
(@SQLInsert(callable=true, ...)).
To check that the execution happens correctly, Hibernate allows you to define one of those three strategies:
• NONE: no check is performed: the store procedure is expected to fail upon issues
Hibernate 3.3.0.GA 45
Entity Beans
• PARAM: like COUNT but using an output parameter rather that the standard mechanism
To define the result check style, use the check parameter (@SQLUpdate(check=ResultCheckStyle.COUNT,
...)).
You can also override the SQL load statement by a native SQL query or a HQL query. You just have to refer to
a named query with the @Loader annotation.
You can use the exact same set of annotations to override the collection related statements.
@OneToMany
@JoinColumn(name="chaos_fk")
@SQLInsert( sql="UPDATE CASIMIR_PARTICULE SET chaos_fk = ? where id = ?")
@SQLDelete( sql="UPDATE CASIMIR_PARTICULE SET chaos_fk = null where id = ?")
private Set<CasimirParticle> particles = new HashSet<CasimirParticle>();
The parameters order is important and is defined by the order Hibernate handle properties. You can see the ex-
pected order by enabling debug logging for the org.hibernate.persister.entity level. With this level en-
abled Hibernate will print out the static SQL that is used to create, update, delete etc. entities. (To see the ex-
pected sequence, remember to not include your custom SQL through annotations as that will override the Hi-
bernate generated static sql.)
Overriding SQL statements for secondary tables is also possible using @org.hibernate.annotations.Table
and either (or all) attributes sqlInsert, sqlUpdate, sqlDelete:
@Entity
@SecondaryTables({
@SecondaryTable(name = "`Cat nbr1`"),
@SecondaryTable(name = "Cat2"})
@org.hibernate.annotations.Tables( {
@Table(appliesTo = "Cat", comment = "My cat table" ),
@Table(appliesTo = "Cat2", foreignKey = @ForeignKey(name="FK_CAT2_CAT"), fetch = FetchMode.SELECT,
sqlInsert=@SQLInsert(sql="insert into Cat2(storyPart2, id) values(upper(?), ?)") )
} )
public class Cat implements Serializable {
The previous example also show that you can give a comment to a given table (promary or secondary): This
comment will be used for DDL generation.
2.4.12. Tuplizer
org.hibernate.tuple.Tuplizer, and its sub-interfaces, are responsible for managing a particular representa-
tion of a piece of data, given that representation's org.hibernate.EntityMode. If a given piece of data is
thought of as a data structure, then a tuplizer is the thing which knows how to create such a data structure and
how to extract values from and inject values into such a data structure. For example, for the POJO entity mode,
the correpsonding tuplizer knows how create the POJO through its constructor and how to access the POJO
properties using the defined property accessors. There are two high-level types of Tuplizers, represented by the
org.hibernate.tuple.EntityTuplizer and org.hibernate.tuple.ComponentTuplizer interfaces. En-
tityTuplizers are responsible for managing the above mentioned contracts in regards to entities, while Compon-
entTuplizers do the same for components. Check the Hibernate reference documentation for more informa-
tion.
To define tuplixer in annotations, simply use the @Tuplizer annotation on the according element
@Entity
@Tuplizer(impl = DynamicEntityTuplizer.class)
public interface Cuisine {
Hibernate 3.3.0.GA 46
Entity Beans
@Id
@GeneratedValue
public Long getId();
public void setId(Long id);
@Tuplizer(impl = DynamicComponentTuplizer.class)
public Country getCountry();
public void setCountry(Country country);
Hibernate 3.3.0.GA 47
Chapter Overriding metadata through
XML. Overriding metadata through XML
The primary target for metadata in EJB3 is annotations, but the EJB3 specification provides a way to override
or replace the annotation defined metadata through an XML deployment descriptor. In the current release only
pure EJB3 annotations overriding are supported. If you wish to use Hibernate specific features in some entities,
you'll have to either use annotations or fallback to hbm files. You can of course mix and match annotated entit-
ies and entities describes in hbm files.
The unit test suite shows some additional XML file samples.
You can define one ot more XML files describing your metadata, these files will be merged by the overriding
engine.
You can define global level metadata available for all XML files. You must not define these metadata more
than once per deployment.
<entity-mappings
xmlns="http://java.sun.com/xml/ns/persistence/orm"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/persistence/orm orm_1_0.xsd"
version="1.0">
<persistence-unit-metadata>
<xml-mapping-metadata-complete/>
<persistence-unit-defaults>
<schema>myschema</schema>
<catalog>mycatalog</catalog>
<cascade-persist/>
</persistence-unit-defaults>
</persistence-unit-metadata>
xml-mapping-metadata-complete means that all entity, mapped-superclasses and embeddable metadata should
be picked up from XML (ie ignore annotations).
schema / catalog will override all default definitions of schema and catalog in the metadata (both XML and
annotations).
cascade-persist means that all associations have PERSIST as a cascade type. We recommend you to not use
this feature.
Hibernate 3.3.0.GA 48
Overriding metadata through XML
<entity-mappings (1)
xmlns="http://java.sun.com/xml/ns/persistence/orm"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/persistence/orm orm_1_0.xsd"
version="1.0">
<package>org.hibernate.test.annotations.reflection</package> (2)
<entity class="Administration" access="PROPERTY" metadata-complete="true"> (3)
<table name="tbl_admin"> (4)
<unique-constraint>
<column-name>firstname</column-name>
<column-name>lastname</column-name>
</unique-constraint>
</table>
<secondary-table name="admin2"> (5)
<primary-key-join-column name="admin_id" referenced-column-name="id"/>
<unique-constraint>
<column-name>address</column-name>
</unique-constraint>
</secondary-table>
<id-class class="SocialSecurityNumber"/> (6)
<inheritance strategy="JOINED"/> (7)
<sequence-generator name="seqhilo" sequence-name="seqhilo"/> (8)
<table-generator name="table" table="tablehilo"/> (9)
...
</entity>
<entity class="PostalAdministration">
<primary-key-join-column name="id"/> (10)
...
</entity>
</entity-mappings>
(1) entity-mappings: entity-mappings is the root element for all XML files. You must declare the xml
schema, the schema file is included in the hibernate-annotations.jar file, no internet access will be pro-
cessed by Hibernate Annotations.
(2) package (optional): default package used for all non qualified class names in the given deployment
descriptor file.
(3) entity: desribes an entity.
metadata-complete defines whether the metadata description for this element is complete or not (in other
words, if annotations present at the class level should be considered or not).
An entity has to have a class attribute refering the java class the metadata applies on.
You can overrides entity name through the name attribute, if none is defined and if an @Entity.name is
present, then it is used (provided that metadata complete is not set).
For netadata complete (see below) element, you can define an access (either FIELD or PROPERTY
(default)). For non medatada complete element, if access is not defined, the @Id position will lead posi-
tion, if access is defined, the value is used.
(4) table: you can declare table properties (name, schema, catalog), if none is defined, the java annotation is
used.
You can define one or several unique constraints as seen in the example
(5) secondary-table: defines a secondary table very much like a regular table except that you can define the
primary key / foreign key column(s) through the primary-key-join-column element. On non metadata
complete, annotation secondary tables are used only if there is no secondary-table definition, annota-
tions are ignored otherwise.
(6) id-class: defines the id class in a similar way @IdClass does
Hibernate 3.3.0.GA 49
Overriding metadata through XML
(7) inheritance: defines the inheritance strategy (JOINED, TABLE_PER_CLASS, SINGLE_TABLE), Available only
at the root entity level
(8) sequence-generator: defines a sequence generator
(9) table-generator: defines a table generator
(10) primary-key-join-column: defines the primary key join column for sub entities when JOINED inheritance
strategy is used
<entity-mappings
xmlns="http://java.sun.com/xml/ns/persistence/orm"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/persistence/orm orm_1_0.xsd"
version="1.0">
<package>org.hibernate.test.annotations.reflection</package>
<entity class="Music" access="PROPERTY" metadata-complete="true">
<discriminator-value>Generic</discriminator-value> (1)
<discriminator-column length="34"/>
...
</entity>
<entity class="PostalAdministration">
<primary-key-join-column name="id"/>
<named-query name="adminById"> (2)
<query>select m from Administration m where m.id = :id</query>
<hint name="org.hibernate.timeout" value="200"/>
</named-query>
<named-native-query name="allAdmin" result-set-mapping="adminrs"> (3)
<query>select *, count(taxpayer_id) as taxPayerNumber
from Administration, TaxPayer
where taxpayer_admin_id = admin_id group by ...</query>
<hint name="org.hibernate.timeout" value="200"/>
</named-native-query>
<sql-result-set-mapping name="adminrs"> (4)
<entity-result entity-class="Administration">
<field-result name="name" column="fld_name"/>
</entity-result>
<column-result name="taxPayerNumber"/>
</sql-result-set-mapping>
<attribute-override name="ground"> (5)
<column name="fld_ground" unique="true" scale="2"/>
</attribute-override>
<association-override name="referer">
<join-column name="referer_id" referenced-column-name="id"/>
</association-override>
...
</entity>
</entity-mappings>
(1) discriminator-value / discriminator-column: defines the discriminator value and the column hold-
ing it when the SINGLE_TABLE inheritance strategy is chosen
(2) named-query: defines named queries and possibly the hints associated to them. Those definitions are ad-
ditive to the one defined in annotations, if two definitions have the same name, the XML one has priority.
(3) named-native-query: defines an named native query and its sql result set mapping. Alternatively, you
can define the result-class. Those definitions are additive to the one defined in annotations, if two
definitions have the same name, the XML one has priority.
(4) sql-result-set-mapping: describes the result set mapping structure. You can define both entity and
column mappings. Those definitions are additive to the one defined in annotations, if two definitions have
the same name, the XML one has priority
(5) attribute-override / association-override: defines a column or join column overriding. This over-
riding is additive to the one defined in annotations
Hibernate 3.3.0.GA 50
Overriding metadata through XML
You can of course defines XML overriding for properties. If metadata complete is defined, then additional
properties (ie at the Java level) will be ignored. Otherwise, once you start overriding a property, all annotations
on the given property are ignored. All property level metadata behave in entity/attributes, mapped-su-
perclass/attributes or embeddable/attributes.
<attributes>
<id name="id">
<column name="fld_id"/>
<generated-value generator="generator" strategy="SEQUENCE"/>
<temporal>DATE</temporal>
<sequence-generator name="generator" sequence-name="seq"/>
</id>
<version name="version"/>
<embedded name="embeddedObject">
<attribute-override name"subproperty">
<column name="my_column"/>
</attribute-override>
</embedded>
<basic name="status" optional="false">
<enumerated>STRING</enumerated>
</basic>
<basic name="serial" optional="true">
<column name="serialbytes"/>
<lob/>
</basic>
<basic name="terminusTime" fetch="LAZY">
<temporal>TIMESTAMP</temporal>
</basic>
</attributes>
You can override a property through id, embedded-id, version, embedded and basic. Each of these elements
can have subelements accordingly: lob, temporal, enumerated, column.
You can define XML overriding for associations. All association level metadata behave in entity/attributes,
mapped-superclass/attributes or embeddable/attributes.
<attributes>
<one-to-many name="players" fetch="EAGER">
<map-key name="name"/>
<join-column name="driver"/>
<join-column name="number"/>
</one-to-many>
<many-to-many name="roads" target-entity="Administration">
<order-by>maxSpeed</order-by>
<join-table name="bus_road">
<join-column name="driver"/>
<join-column name="number"/>
<inverse-join-column name="road_id"/>
<unique-constraint>
<column-name>driver</column-name>
<column-name>number</column-name>
</unique-constraint>
</join-table>
</many-to-many>
<many-to-many name="allTimeDrivers" mapped-by="drivenBuses">
</attributes>
You can override an association through one-to-many, one-to-one, many-to-one, and many-to-many. Each of
these elements can have subelements accordingly: join-table (which can have join-columns and inverse-
Hibernate 3.3.0.GA 51
Overriding metadata through XML
join-columns), join-columns, map-key, and order-by. mapped-by and target-entity can be defined as attrib-
utes when it makes sense. Once again the structure is reflects the annotations structure. You can find all se-
mantic informations in the chapter describing annotations.
Hibernate 3.3.0.GA 52
Chapter 4. Additional modules
Hibernate Annotations mainly focus on persistence metadata. The project also have a nice integration with two
Hibernate modules.
4.1.1. Description
Annotations are a very convenient and elegant way to specify invariant constraints for a domain model. You
can, for example, express that a property should never be null, that the account balance should be strictly posit-
ive, etc. These domain model constraints are declared in the bean itself by annotating its properties. A validator
can then read them and check for constraint violations. The validation mechanism can be executed in different
layers in your application without having to duplicate any of these rules (presentation layer, data access layer).
Following the DRY principle, Hibernate Validator has been designed for that purpose.
Hibernate Validator works at two levels. First, it is able to check in-memory instances of a class for constraint
violations. Second, it can apply the constraints to the Hibernate metamodel and incorporate them into the gener-
ated database schema.
Each constraint annotation is associated to a validator implementation responsible for checking the constraint
on the entity instance. A validator can also (optionally) apply the constraint to the Hibernate metamodel, allow-
ing Hibernate to generate DDL that expresses the constraint. With the appropriate event listener, you can ex-
ecute the checking operation on inserts and updates done by Hibernate. Hibernate Validator is not limited to use
with Hibernate. You can easily use it anywhere in your application.
When checking instances at runtime, Hibernate Validator returns information about constraint violations in an
array of InvalidValue s. Among other information, the InvalidValue contains an error description message
that can embed the parameter values bundle with the annotation (eg. length limit), and message strings that may
be externalized to a ResourceBundle .
If Hibernate Validator (hibernate-validator.jar) is available in the classpath, Hibernate Annotations will in-
tegrates in two ways:
• Constraints will be applied to the Data Definition Language. In other words, the database schema will re-
flect the constraints (provided that you use the hbm2ddl tool).
• Before an entity change is applied to the database (insert or update), the entity is validated. Validation er-
rors, if any, will be carried over through an InvalidStateException.
For entities free of validation rules, the runtime performance cost is null.
Hibernate 3.3.0.GA 53
Additional modules
4.2.1. Decription
Full text search engines like Apache Lucene™ are a very powerful technology to bring free text/efficient quer-
ies to applications. If suffers several mismatches when dealing with a object domain model (keeping the index
up to date, mismatch between the index structure and the domain model, querying mismatch...) Hibernate
Search indexes your domain model thanks to a few annotations, takes care of the database / index synchroniza-
tion and brings you back regular managed objects from free text queries. Hibernate Search is using Apache Lu-
cene [http://lucene.apache.org] under the cover.
Hibernate Search integrates with Hibernate Annotations transparently provided that hibernate-search.jar is
present in the classpath. If you do not wish to autoregister Hibernate Search event listeners, you can set hi-
bernate.search.autoregister_listeners to false. Such a need is very uncommon and not recommanded.
Hibernate 3.3.0.GA 54