| ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
| Castor JDO - Advanced features
Documentation Author(s): Reference: Introduction Caching Dependent and related relationships Different cardinalities of relationship Lazy Loading 1:1 relations 1:M and M:N relations Multiple columns primary keys Callback interface for persistent operations IntroductionAs explained at the introduction to Castor JDO, Castor has support for many advanced features such as caching, depend relations, inheritance, polymorphism, etc. The below sections detail these features, as their understanding is required to use Castor JDO in a performant and secure way. CachingAll information related to caching and related concepts supported by Castor has been consolidated into one place, and is available here. Dependent and related relationshipsCastor distinguishes the relationship of two objects as dependent or related, and maintains the life cycle independently for the two types of relationships. Starting from Castor 0.9, the developer can explicitly define a dependent relationship in the mapping file. When using independent relations, related objects life cycle is independent of each other, meaning that they have to be created, removed and updated (for long transaction) independently. When using dependent relations, one data object class must be declared as depends on one other data object class in the mapping file, and such an object is called a dependent data object class. A data object class without depends declared in the mapping is called a master object. A master object can be depended upon by zero or more dependent data object class. As of Castor 0.9, a dependent object class can be related to other master data object classes including extended classes, but cannot depend on more than one master class. If an object class declared as depends on another class, it may not be created, removed or updated separately. Attempting to create, remove or update a dependent object will result in ObjectNotPersistcapableException. Note that Castor doesn't allow a dependent object instance to change its master object instance during a transaction. Each dependent object can have only one master object. Both dependent and master objects must have identities, and may or may not make use of key-generators.
Here is the DTD for declaring dependent object:
For example,
Different cardinalities of relationshipCastor supports different cardinalities of relationship, namely one-to-one, one-to-many, and many-to-many. Many-to-many relationship must be related rather than dependent, because each dependent object can have only one master object. Many-to-many requires a separate table for storing the relations between two types of objects. Many-to-many introduces two attributes, namely many-key and many-table that reside in the <sql> element which is a sub-element of the <field> element. For all many-to-many relations, a many-table must be specified. If the column name of the primary key of the class is different from the foreign keys columns of the class in the relation tables, then the relation table columns can be specified using the many-key attributes. Similarly, if the column name of the primary key of the related class is different from the foreign key columns of the related class, then the relation table columns can be specified using the name attribute. The many-table is used to store relations in a separate table
So, for example, if the SQL table is the following,
Then, the mapping for employee data object would look like this
Lazy LoadingAs of release 0.9.6, Castor has full support for lazy loading object instances referenced as part of all relation types currently supported:
1:1 relationsAs of release 0.9.6, Castor supports lazy-loading of 1:1 relations. Imagine the following class mapping:
Per definition, when an instance of Department is loaded through e.g. Database.load(), Castor will not (pre-)load the Employee instance referenced (as such reducing the size pf the initial query as well as the size of the result set returned). Only when the Emplyoee instance is accessed through Department.getEmployee(), Castor will load the actual object into memory from the persistence store. This means that if the Employee instance is not accessed at all, not only will the initial query to load the Department object have had its complexity reduced, but no performance penalty will be incurred for the additional access to the persistence store either. 1:M and M:N relationsThe elements in the collection are only loaded when the application asks for the object from the collection, using, for example, iterator.next(). The iterator in Castors lazy collection is optimized to return a loaded object first. In the mapping file, lazy loading is specified in the element of the collections <field>, for example,
declares that the collection of type Employee in a Department is lazy loaded. If lazy loading is specified for a field of a class, Castor will set the field with a special collection which contains only the identities of the objects. Because of that, it requires the data object to have the method setDepartment( Collection department) in the data object class which was not required in previous versions.
Multiple columns primary keys
The support of multiple column primary keys (also called compound primary keys) was another major enhancement added into Castor 0.9. Specifying multiple column primary keys is simple and straightforward, in the mapping file,
Multiple column primary keys work with both master and dependent objects, all cardinalities of relationship, including one-to-one, one-to-many and many-to-many, as well as lazy loading. However, multiple column primary keys should only be used to adhere to an existing database design, not when designing a new database. In general, it is not a good idea to use an identity or identities which can be modified by the user, or which contain application-visible data. For example, if the system allows the user name to be changed, using user name as identity is highly discouraged, as this practice can require a major data migration to a new schema to update all foreign keys to adhere to a new primary key structure, should the user name no longer be adequate as a primary key. It should be noted that Castor doesnt support identity change, as specified in the ODMG 3.0 specification. So, primary keys changes are almost certainly a large trade off between data integrity and performance. Well chosen primary keys are usually single (not multiple) column numeric or character fields for the reasons outlined above, as well as performance, as joining operations are faster for single column primary keys. Callback interface for persistent operationsFor the various persistence operations as available through the Database interface, Castor JDO provides a callback interface that informs the implementing class on events taking place related to selected persistence operations. Once your entity class implements the Persistence interface, you'll have to provide implementations for the following methods (with their respective semantics described next to them):
| |||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||