alvinalexander.com | career | drupal | java | mac | mysql | perl | scala | uml | unix  

Hibernate example source code file (Locking.xml)

This example Hibernate source code file (Locking.xml) is included in the DevDaily.com "Java Source Code Warehouse" project. The intent of this project is to help you "Learn Java by Example" TM.

Java - Hibernate tags/keywords

a, for, hibernate, hibernate, if, if, java, optional, optional, the, the, this, update, xml

The Hibernate Locking.xml source code

<?xml version='1.0' encoding='utf-8' ?>
<!DOCTYPE chapter PUBLIC "-//OASIS//DTD DocBook XML V4.5//EN" "http://www.oasis-open.org/docbook/xml/4.5/docbookx.dtd" [
<!ENTITY % BOOK_ENTITIES SYSTEM "Hibernate_Development_Guide.ent">
%BOOK_ENTITIES;
]>
<chapter>
  <title>Locking
  <para>
    Locking refers to actions taken to prevent data in a relational database from changing between the time it is read
    and the time that it is used.
  </para>
  <para>
    Your locking strategy can be either <firstterm>optimistic or pessimistic.
  </para>
  <variablelist>
    <title>Locking strategies
    <varlistentry>
      <term>Optimistic
      <listitem>
        <para>
          Optimistic locking ssumes that multiple transactions can complete without affecting each other, and that
          therefore transactions can proceed without locking the data resources that they affect. Before committing,
          each transaction verifies that no other transaction has modified its data. If the check reveals conflicting
          modifications, the committing transaction rolls back<footnote>.
        </para>
      </listitem>
    </varlistentry>
    <varlistentry>
      <term>Pessimistic
      <listitem>
        <para>
          Pessimistic locking assumes that concurrent transactions will conflict with each other, and requires resources
          to be locked after they are read and only unlocked after the application has finished using the data.
        </para>
      </listitem>
    </varlistentry>
  </variablelist>
  <para>
    Hibernate provides mechanisms for implementing both types of locking in your applications.
  </para>
  <section>
    <title>Optimistic
    <para>
      When your application uses long transactions or conversations that span several database transactions, you can
      store versioning data, so that if the same entity is updated by two conversations, the last to commit changes is
      informed of the conflict, and does not override the other conversation's work. This approach guarantees some
      isolation, but scales well and works particularly well in <firstterm>Read-Often Write-Sometimes
      situations.
    </para>
    <para>
      Hibernate provides two different mechanisms for storing versioning information, a dedicated version number or a
      timestamp.
    </para>
    <variablelist>
      <varlistentry>
        <term>Version number
        <listitem>
          <para>

          </para>
        </listitem>
      </varlistentry>
      <varlistentry>
        <term>Timestamp
        <listitem>
          <para>

          </para>
        </listitem>
      </varlistentry>
    </variablelist>
    <note>
      <para>
        A version or timestamp property can never be null for a detached instance. Hibernate detects any instance with a
        null version or timestamp as transient, regardless of other unsaved-value strategies that you specify. Declaring
        a nullable version or timestamp property is an easy way to avoid problems with transitive reattachment in
        Hibernate, especially useful if you use assigned identifiers or composite keys.
      </para>
    </note>
    
    <section>
      <title>Dedicated version number
      <para>
        The version number mechanism for optimistic locking is provided through a <literal>@Version
        annotation.
      </para>
      <example>
        <title>The @Version annotation
        <programlisting language="Java" role="JAVA">
        <para>
          Here, the version property is mapped to the <literal>OPTLOCK column, and the entity manager uses it
          to detect conflicting updates, and prevent the loss of updates that would be overwritten by a
          <firstterm>last-commit-wins strategy.
        </para>
      </example>
      <para>
        The version column can be any kind of type, as long as you define and implement the appropriate
        <classname>UserVersionType.
      </para>
      <para>
        Your application is forbidden from altering the version number set by Hibernate. To artificially increase the
        version number, see the documentation for properties
        <property>LockModeType.OPTIMISTIC_FORCE_INCREMENT or
        <property>LockModeType.PESSIMISTIC_FORCE_INCREMENTcheck in the Hibernate Entity Manager reference
        documentation.
      </para>
      <note>
        <title>Database-generated version numbers
        <para>
          If the version number is generated by the database, such as a trigger, use the annotation
          <literal>@org.hibernate.annotations.Generated(GenerationTime.ALWAYS).
        </para>
      </note>
      <example>
        <title>Declaring a version property in hbm.xml
        <programlisting language="XML" role="XML">
        <informaltable>
          <tgroup cols="2">
            <tbody>
              <row>
                <entry>column
                <entry>The name of the column holding the version number. Optional, defaults to the property
                name. </para>
              </row>
              <row>
                <entry>name
                <entry>The name of a property of the persistent class.
              </row>
              <row>
                <entry>type
                <entry>The type of the version number. Optional, defaults to
                <literal>integer.
              </row>
              <row>
                <entry>access
                <entry>Hibernate's strategy for accessing the property value. Optional, defaults to
                <literal>property.
              </row>
              <row>
                <entry>unsaved-value
                <entry>Indicates that an instance is newly instantiated and thus unsaved. This distinguishes it
                from detached instances that were saved or loaded in a previous session. The default value,
                <literal>undefined, indicates that the identifier property value should be
                used. Optional.</para>
              </row>
              <row>
                <entry>generated
                <entry>Indicates that the version property value is generated by the database. Optional, defaults
                to <literal>never.
              </row>
              <row>
                <entry>insert
                <entry>Whether or not to include the version column in SQL insert
                statements. Defaults to <literal>true, but you can set it to false if the
                database column is defined with a default value of <literal>0.
              </row>
            </tbody>
          </tgroup>
        </informaltable>
      </example>
    </section>
    
    <section>
      <title>Timestamp
      <para>
        Timestamps are a less reliable way of optimistic locking than version numbers, but can be used by applications
        for other purposes as well. Timestamping is automatically used if you the <code>@Version annotation on a
        <type>Date or Calendar.
      </para>
      <example>
        <title>Using timestamps for optimistic locking
        <programlisting language="Java" role="JAVA">
      </example>
      <para>
        Hibernate can retrieve the timestamp value from the database or the JVM, by reading the value you specify for
        the <code>@org.hibernate.annotations.Source annotation. The value can be either
        <literal>org.hibernate.annotations.SourceType.DB or
        <literal>org.hibernate.annotations.SourceType.VM. The default behavior is to use the database, and is
        also used if you don't specify the annotation at all.
      </para>
      <para>
        The timestamp can also be generated by the database instead of Hibernate, if you use the
        <code>@org.hibernate.annotations.Generated(GenerationTime.ALWAYS) annotation.
      </para>
      <example>
        <title>The timestamp element in hbm.xml
        <programlisting language="XML" role="XML">
        <informaltable>
          <tgroup cols="2">
            <tbody>
              <row>
                <entry>column
                <entry>The name of the column which holds the timestamp. Optional, defaults to the property
                namel</para>
              </row>
              <row>
                <entry>name
                <entry>The name of a JavaBeans style property of Java type Date or Timestamp of the persistent
                class.</para>
              </row>
              <row>
                <entry>access
                <entry>The strategy Hibernate uses to access the property value. Optional, defaults to
                <literal>property.
              </row>
              <row>
                <entry>unsaved-value A version property which indicates than instance is newly
                instantiated, and unsaved. This distinguishes it from detached instances that were saved or loaded in a
                previous session. The default value of <literal>undefined indicates that Hibernate uses the
                identifier property value.</para>
              </row>
              <row>
                <entry>source
                <entry>Whether Hibernate retrieves the timestamp from the database or the current
                JVM. Database-based timestamps incur an overhead because Hibernate needs to query the database each time
                to determine the incremental next value. However, database-derived timestamps are safer to use in a
                clustered environment. Not all database dialects are known to support the retrieval of the database's
                current timestamp. Others may also be unsafe for locking, because of lack of precision.</para>
              </row>
              <row>
                <entry>generated
                <entry>Whether the timestamp property value is generated by the database. Optional, defaults to
                <literal>never.
              </row>
            </tbody>
          </tgroup>
        </informaltable>
      </example>
    </section>

  </section>
  
  <section>
    <title>Pessimistic
    <para>
      Typically, you only need to specify an isolation level for the JDBC connections and let the database handle
      locking issues. If you do need to obtain exclusive pessimistic locks or re-obtain locks at the start of a new
      transaction, Hibernate gives you the tools you need.
    </para>
    <note>
      <para>
        Hibernate always uses the locking mechanism of the database, and never lock objects in memory.
      </para>
    </note>
    <section>
      <title>The LockMode class
      <para>
        The <classname>LockMode class defines the different lock levels that Hibernate can acquire.
      </para>
      <informaltable>
        <tgroup cols="2">
          <tbody>
            <row>
              <entry>LockMode.WRITE
              <entry>acquired automatically when Hibernate updates or inserts a row.
            </row>
            <row>
              <entry>LockMode.UPGRADE
              <entry>acquired upon explicit user request using SELECT ... FOR UPDATE on databases
              which support that syntax.</para>
            </row>
            <row>
              <entry>LockMode.UPGRADE_NOWAIT
              <entry>acquired upon explicit user request using a SELECT ... FOR UPDATE NOWAIT in
              Oracle.</para>
            </row>
            <row>
              <entry>LockMode.READ
              <entry>acquired automatically when Hibernate reads data under Repeatable Read or
              <phrase>Serializable isolation level. It can be re-acquired by explicit user
              request.</para>
            </row>
            <row>
              <entry>LockMode.NONE
              <entry>The absence of a lock. All objects switch to this lock mode at the end of a
              Transaction. Objects associated with the session via a call to <methodname>update() or
              <methodname>saveOrUpdate() also start out in this lock mode. 
            </row>
          </tbody>
        </tgroup>
      </informaltable>
      <para>
        The explicit user request mentioned above occurs as a consequence of any of the following actions:
      </para>
      <itemizedlist>
        <listitem>
          <para>
            A call to <methodname>Session.load(), specifying a LockMode.
          </para>
        </listitem>
        <listitem>
          <para>
            A call to <methodname>Session.lock().
          </para>
        </listitem>
        <listitem>
          <para>
            A call to <methodname>Query.setLockMode().
          </para>
        </listitem>
      </itemizedlist>
      <para>
        If you call <methodname>Session.load() with option  or
        <option>UPGRADE_NOWAIT, and the requested object is not already loaded by the session, the object is
        loaded using <code>SELECT ... FOR UPDATE. If you call load() for an object that
        is already loaded with a less restrictive lock than the one you request, Hibernate calls
        <methodname>lock() for that object.
      </para>
      <para>
        <methodname>Session.lock() performs a version number check if the specified lock mode is
        <literal>READ, UPGRADE, or UPGRADE_NOWAIT. In the case of
        <literal>UPGRADE or UPGRADE_NOWAIT, SELECT ... FOR UPDATE syntax is
        used.
      </para>
      <para>
        If the requested lock mode is not supported by the database, Hibernate uses an appropriate alternate mode
        instead of throwing an exception. This ensures that applications are portable.
      </para>
    </section>
  </section>
</chapter>

Other Hibernate examples (source code examples)

Here is a short list of links related to this Hibernate Locking.xml source code file:

... this post is sponsored by my books ...

#1 New Release!

FP Best Seller

 

new blog posts

 

Copyright 1998-2024 Alvin Alexander, alvinalexander.com
All Rights Reserved.

A percentage of advertising revenue from
pages under the /java/jwarehouse URI on this website is
paid back to open source projects.