resin-ee CMP Inheritance

Amber supports database-backed inheritance, allowing for persistent-backed polymorphism and more sophisticated object-oriented modelling.

The example uses a single table to represent both Student and Prefect values. The "type" column serves as a discriminator to select the proper type.

tutorial query
resin-web.xml configuration META-INF/persistence.xml configuration The student bean The prefect bean The course servlet
CREATE TABLE amber_inherit_student ( id BIGINT PRIMARY KEY auto_increment, type VARCHAR(10), name VARCHAR(250) ); INSERT INTO amber_inherit_student VALUES(1, 'student', 'Harry Potter') INSERT INTO amber_inherit_student VALUES(2, 'prefect', 'Ron Weasley') INSERT INTO amber_inherit_student VALUES(1, 'prefect', 'Hermione Granger')

The example has two beans to illustrate inheritance. The Student bean is the base class and represents all students. Some students are also prefects. The prefect students are represented by a Prefect class.

In the database, the type column distinguishes between Students and Prefects. A type of "student" creates a Student bean and a type of "prefect" creates a Prefect bean.

package example; @javax.ejb.Entity(access=FIELD) @javax.ejb.Table(name="ejb3_inherit_student") @javax.ejb.Inheritance(discriminatorValue="student") @javax.ejb.DiscriminatorColumn(name="type") public class Student { @javax.ejb.Id(generator=AUTO) @javax.ejb.Column(name="id") private long _id; @javax.ejb.Basic @javax.ejb.Column(name="name") private String _name; public Student() { } public Student(String name) { _name = name; } public String getName() { return _name; } public String toString() { return "Student[_name]"; } }

The @Inheritance annotation marks the entity bean as supporting inheritance.

Amber supports two inheritance types: SINGLE_TABLE and JOINED. SINGLE_TABLE uses a single table for all classes and JOINED adds additional tables for child classes. Both types use a discriminator column in the base table to mark the Java type of the database entry. SINGLE_TABLE is the default.

This example uses the default SINGLE_TABLE since there's no additional information for the Prefect. The example only uses inheritance to illustrate the database polymorphism.

The values in the discriminator column determine the Java class. Each entity will specify its discriminator value in the discriminatorValue annotation. In the example, the Student class has a "student" value and the Prefect class has a "prefect" value.

Both SINGLE_TABLE and JOINED use a discriminator column to distinguish the classes. The @DiscriminatorColumn annotation specifies the column name. In this example, the column is "type". The default discriminator column is "TYPE".

package example; import static javax.ejb.AccessType.FIELD; @javax.ejb.Entity(access=FIELD) @javax.ejb.Inheritance(discriminatorValue="prefect") public class Prefect { public Prefect() { } public Prefect(String name) { super(name); } public String toString() { return "Prefect[" + getName() + "]"; } }

In this example, the Prefect class has no additional columns, so it only specifies the @Entity and @Inheritance values. When Resin loads the objects from the database, it will instantiate the proper type.

The client code using inheritance is no different from normal queries. The Query will perform any necessary database joins to return the proper Java class. In this case, prefects will return Prefect objects and students will return Student objects.

public void service(PrintWriter out) throws IOException { out.println("<h3>Students</h3>"); Query query = _manager.createQuery("SELECT o FROM Student o"); for (Object student : query.listResults()) { out.println("student: " + student() + "<br>"); } } } <h3>Students</h3> Student[Harry Potter] Prefect[Hermione Granger] Prefect[Ron Weasley]