Introduction The Mapping File The <class> element The <map-to> element The <field> element The <sql> element Usage Pattern
Introduction The Castor mapping file also provides a mechanism for binding a Java object model to a relational database model. This is usually referred to as object-to-relational mapping (O/R mapping). O/R mapping bridges the gap between an object model and a relational model. The mapping file provides a description of the Java object model to Castor JDO. Via Castor XML, Castor JDO uses the information in the mapping file to map Java objects to relational database tables. The following is a high-level example of a mapping file:
<mapping>
<class>
<map-to />
<field>
<sql />
</field>
...
</class>
</mapping>
Each Java object is represented by a <class> element composed of attributes, a <map-to> element and <field> elements. The <map-to> element contains a reference to the relational table to which the Java object maps. Each <field> element represents either a public class variable or a the variable's accessor/mutator methods (depending upon the mapping info). Each <field> element is composed of attributes and one <sql> element. The <sql> element represents the field in the relational table to which the <field> element maps. It is possible to use the mapping file and Castor's default behavior in conjunction. When Castor handles an object but is unable to locate information about it in the mapping file, it will rely upon its default behavior. Castor makes use of the Java programming language Reflection API to introspect the Java objects to determine the methods to use. This is the reason some attributes are not required in the mapping file. The Mapping File The <class> element
<!ELEMENT class ( description?, cache-type?, map-to?, field+ )>
<!ATTLIST class
name ID #REQUIRED
extends IDREF #IMPLIED
depends IDREF #IMPLIED
auto-complete ( true |false ) "false"
identity CDATA #IMPLIED
access ( read-only | shared | exclusive | db-locked ) "shared"
key-generator IDREF #IMPLIED >
The <class> element contains all the information used to map a Java object to a relational database. The content of <class> is mainly used to describe the fields that will be mapped. Description of the attributes: | - | name: The fully qualified package name of the Java object to map to. | - | extends: Should be used _only_ if this Java object extends another Java object for which mapping information is provided. It should _not_ be used if the extended Java object is not referenced in the mapping file. | - | depends: For more information on this field, please see Dependent and related relationships. | - | identity: For more information on this field, please see Design -> Persistence. | - | access: For more information on this field, please see Locking Modes. | - | key-generator: For more information on this field, please see KeyGen. | Description of the elements: | - | <description>: An optional description. | - | <cache-type>: For more information on this field please see Bounded Dirty Checking and LRU Cache. | - | <map-to>: Used to tell Castor the name of the relational table to which to map. | - | <field>: Zero or more <field> elements are used to describe properties of each Java object. | The <map-to> element
<!ELEMENT map-to EMPTY>
<!ATTLIST map-to
table NMTOKEN #IMPLIED
xml NMTOKEN #IMPLIED
ns-uri NMTOKEN #IMPLIED
ns-prefix NMTOKEN #IMPLIED
ldap-dn NMTOKEN #IMPLIED
ldap-oc NMTOKEN #IMPLIED>
<map-to> is used to specify the name of the item that should be associated with the given Java object. The <map-to> element is only used for the root Java object. Description of the attributes: | - | table: The name of the relational database table to which the Java object is associated. | The <field> element
<!ELEMENT field ( description?, sql?, xml?, ldap? )>
<!ATTLIST field
name NMTOKEN #REQUIRED
type NMTOKEN #IMPLIED
required ( true | false ) "false"
direct ( true | false ) "false"
lazy ( true | false ) "false"
transient ( true | false ) "false"
get-method NMTOKEN #IMPLIED
set-method NMTOKEN #IMPLIED
create-method NMTOKEN #IMPLIED
collection ( array | vector | hashtable | collection | set | map ) #IMPLIED>
The <field> element is used to describe a property of a Java object. It provides: | - | the identity ('name') of the property | - | the type of the property (inferred from 'type' and 'collection') | - | the access method of the property (inferred from 'direct', 'get-method', 'set-method') | From this information, Castor is able to access a given property in the Java object. In order to determine the signature that Castor expects, there are two easy rules to apply. 1. Determine <type>. | - | If there is no 'collection' attribute, the object type is the value of the 'type' attribute. The value of the type attribute can be a fully qualified Java object like 'java.lang.String' or one of the allowed aliases: short name | Primitive type? | Java Class | big-decimal | N | java.math.BigDecimal | boolean | Y | java.lang.Boolean.TYPE | byte | Y | java.lang.Byte.TYPE | bytes | N | byte[] | char | Y | java.lang.Character.TYPE | chars | N | char[] | clob | N | java.sql.Clob | date | N | java.util.Date | double | Y | java.lang.Double.TYPE | float | Y | java.lang.Float.TYPE | integer | Y | java.lang.Integer.TYPE | locale | N | java.util.Locale | long | Y | java.lang.Long.TYPE | other | N | java.lang.Object | short | Y | java.lang.Short.TYPE | string | N | java.lang.String | strings | N | String[] | stream | N | java.io.InputStream | Castor will try to cast the data in the mapping file to the proper Java type. | - | If there is a collection attribute, the items in the following table can be used: name | type | default implementation | array | <type_attribute>[] | <type_attribute>[] | vector | java.util.Vector | java.util.Vector | hashtable | java.util.Hashtable | java.util.Hashtable | collection | java.util.Collection | java.util.ArrayList | arraylist | java.util.ArrayList | java.util.ArrayList | set | java.util.Set | java.util.HashSet | map | java.util.Map | java.util.HashMap | The type of the object inside the collection is the 'type' attribute. The 'default implementation' is the type used if the object holding the collection is found to be null and needs to be instantiated. For hashtable and map, Castor will add an object using the put(object, object) method - the object is both the key and the value. This will be improved in the future. | It is necessary to use a collection when the content model of the element expects more than one element of the specified type. This is how the 'to-many' portion of a relationship is described. 2. Determine the signature of the method If 'direct' is set to true, Castor expects to find a public Java object variable with the given signature:
public <type> <name>;
If 'direct' is set to false or omitted, Castor will access the property though accessor methods. Castor determines the signature of the accessors and mutators as follows: If the 'get-method' or 'set-method' attributes are supplied, it will try to find a function with the following signature:
public <type> <get-method>();
or
public void <set-method>(<type> value);
If 'get-method' or 'set-method' attributes are not provided, Castor will try to find the following function:
public <type> get<capitalized-name>();
or
public void set<capitalized-name>(<type> value);
<capitalized-name> means that Castor uses the <name> attribute by changing its first letter to uppercase without modifying the other letters. The content of the <field> element will contain the information about how to map this field to the relational table. Description of the attributes: | - | name: If 'direct' access is used, 'name' should be the name of a public variable in the object we are mapping (the field must be public, not static and not transient). If no direct access and no 'get-/set-method' is specified, this name will be used to infer the name of the accessor and mutator methods. | - | type: The Java type of the field. This is used to access the field. Castor will use this information to cast the data type(e.g. string into integer). It is also used to define the signature of the accessor and mutator methods. If a collection is specified, this is used to specify the type of the object inside the collection. See description above for more details. | - | required: If true, the field is not optional. | - | direct: If true, Castor expects a public variable in the object and will modify it directly. | - | collection: If a parent object expects more than one occurrence of one of its fields, it is necessary to specify which collection type Castor will use to handle them. The type specified is used to define the type of the content inside the collection. | - | get-method: An optional name of the accessor method Castor should use. If this attribute is not set, Castor will try to guess the name with the algorithm described above. | - | set-method: An optional name of the mutator method Castor should use. If this attribute is not set, Castor will try to guess the name with the algorithm described above. | - | create-method: Factory method for instantiation of the object. | The <sql> element
<!ELEMENT sql EMPTY>
<!ATTLIST sql
name NMTOKENS #IMPLIED
type NMTOKENS #IMPLIED
many-key NMTOKENS #IMPLIED
many-table NMTOKEN #IMPLIED
read-only ( true | false ) "false"
dirty ( check | ignore ) "check">
The <sql> element is used to denote information about the database table to which a Java object is mapped. It should be declared for all <field> elements. Each <field> element contains one <sql> element. The <sql> element correlates directly to the <map-to> element for the containing <class> element. The <sql> elements contains the following attributes: | - | name: The name of the column in the database table. | - | type: The JDBC type of the column. It is inferred from the object when the type of this field is a persistent Java class that is defined elsewhere in the mapping. | - | many-key: Specifies the name of the column that holds the foreign key to this object. That column is in the database table that stores objects of the Java type of this field. Used only for 'to-many' relations. | - | many-table: Specifies the name of the bridge table that contains the primary keys of the object on each side of the relationship. This is only used for many-to-many relationships. | - | read-only: If true, the column in the relational database table will only be read, not updated or deleted. | - | dirty: If the value is 'ignore', the field will not be checked against the database for modification. | Usage Pattern Here is an example of a mapping file and the corresponding Java object and DDL for the databse table. The following is an example Java object:
package myapp;
public class Product
{
private int _id;
private String _name;
private float _price;
private ProductGroup _group;
public int getId()
...
public void setId( int anId )
...
public String getName()
...
public void setName( String aName )
...
public float getPrice()
...
public void setPrice( float aPrice )
...
public ProductGroup getProductGroup()
...
public void setProductGroup( ProductGroup aProductGroup )
...
}
| The following is the relational database table:
create table prod
(
id int not null,
name varchar(200) not null,
price numeric(18,2) not null,
group_id int not null
);
| The following is the mapping file for the example Java object:
<?xml version="1.0"?>
<!DOCTYPE mapping PUBLIC "-//EXOLAB/Castor Object Mapping DTD Version 1.0//EN"
"http://castor.exolab.org/mapping.dtd">
<mapping>
<class name="myapp.Product" identity="id">
<map-to table="prod" />
<field name="id" type="integer">
<sql name="id" type="integer" />
</field>
<field name="name" type="string">
<sql name="name" type="char" />
</field>
<field name="price" type="float">
<sql name="price" type="numeric" />
</field>
<field name="group" type="myapp.ProductGroup" >
<sql name="group_id" />
</field>
</class>
</mapping>
| |