<generator>是<id>的子元素,该元素的作用是指定主键的生成器。它通过一个class属性指定生成器对应的类,该类必须实现org.hibernate.id.IdentifierGenerator接口。如果需要传递参数,需要通过<param>子元素指定。例如:
 <id name="id" column="ID" type="long">
  <generator class="hilo">
   <param name="table">uid_table</param>
   <param name="column">next_hi_value_column</param>
  </generator>
 </id>

    Hibernate提供了很多内置的生成器实现。下面是一些内置生成器的快捷名称:

(1)assigned算法
    主键由外部程序负责生成,无须Hibernate参与,因此需要应用程序在执行save()之前为对象分配一个标识符。这是<generator>元素没有指定时的默认生成策略。
    当选择assigned生成器时,除非有一个version或timestamp属性,或者你定义了Interceptor.isUnsaved(),否则需要让Hibernate使用unsaved-value="undefined",强制Hibernate查询数据库来确定一个实例是瞬时的(transient)还是托管的(detached)。例:
 <id name="id" column="ID" type="long" unsaved-value="undefined">
  <generator class="assigned"/>
 </id>

(2)hilo算法
    使用hilo(高/低位)算法高效地生成long,short或者int类型的标识符。高/低位算法生成的标识符只在一个特定的数据库中是唯一的,需要额外的数据库表或字段提供高位值来源。
    hilo生成器需要一个“特殊”的数据库表来保存下一个可用的“hi”值。使用该算法是需要通过以下3个参数来指定数据库中的表信息来保存“hi”值:
    table:保存ID的表名。
    column:保存ID的字段名。
    max_lo:最大编号。
例:
 <id name="id" column="ID" type="long">
  <generator class="hilo">
   <param name="table">hi_value</param>
   <param name="columm">next_value</param>
   <param name="max_lo">100</param>
  </generator>
 </id>

(3)seqhilo算法
    与hilo类似,通过hi/lo算法实现的主键生成机制,只是主键历史状态保存在Sequence中,适用于支持Sequence的数据库,如Oracle。
    seqhilo生成器使用一个Oracle风格的序列,这要求数据库支持序列。它需要包括以下两个参数。
    Sequence:序列名。
    max_lo:最大编号。
例:
 <id name="id" column="ID" type="long">
  <generator class="seqhilo">
   <param name="sequence">hi_value</param>
   <param name="max_lo">100</param>
  </generator>
 </id>

(4)increment算法
    increment标识主键按数值顺序递增。此方式的实现机制为,在当前应用实例中维持一个变量以保存当前的最大值,之后每次需要生成主键的时候将些值加1作为主键。
例:
 <id name="id" column="ID" type="long">
  <generator class="increment"/>
 </id>
    这种方式可能产生的问题是:如果当前有多个实例访问同一个数据库,那么由于各个实例各自维护主键状态,不同实例可能生成同样的主键,从而造成主键重复异常。因此,如果同一数据库有多个实例访问,此方式必须避免使用。

(5)identy算法
    identy采用数据库提供的主键生成机制,如DB2、MySQL、Sybase、MS SQL中的主键生成机制,这需要这些数据库内部支持标识字段。
例:
 <id name="id" column="ID" type="long" unsaved-value="0">
  <generator class="identy"/>
 </id>

(6)sequence算法
    sequence采用数据库提供的Sequnce机制生成机制生成主键,如Oracle中的Sequence。对于内部支持序列的数据库(DB2、Oracle、PostgreSQL、Interbase、McKoi、SAP DB),你可以使用Sequence风格的关键字生成。
例:
 <id name="id" column="ID" type="long">
  <generator class="seqence">
   <param name="sequence">user_id_sequence</param>
  </generator>
 </id>
    sequence与seqhilo都使用数据库的序列,但是不同的是,sequence是使用序列生成的ID,而seqhilo是用序列来保存程序生成的ID,该ID由算法自动生成。

(7)native算法
    由Hibernate根据底层数据库自行判断采用identity、hilo和sequence其中一种作为主键生成方式。对于跨平台开发,native策略会从identity、hilo和sequence中进行选择,选择哪一个,这取决于底层数据库的支持能力。
例:
 <id name="id" column="ID" type="long">
  <generator class="native"/>
 </id>

(8)uuid.hex算法
    由Hibernate基于128位唯一值产生算法UUID生成16进制数值(编码后以长度32位的字符串表示)作为主键。
    UUID包含:IP地址、JVM的启动时间(精确到1/4秒)、系统时间和一个计数器值(在JVM中唯一)。在Java代码中不可能获得MAC地址或者内在地址,所以这已经是我们在不使用JNI(Java Native Interface,JAVA本地调用)的前提下能做的最好实现了。
例:
 <id name="id" column="ID" type="long">
  <generator class="uuid.hex"/>
 </id>
    一般而言,利用uuid.hex方式生成主键将提供最好的性能和数据库平台实用性。

(9)uuid.string算法
    与uuid.hex类似,只是生成的主键未进行编码(长度16)。在某些数据库中可能出现问题,如(PostgreSQL)。
例:
 <id name="id" column="ID" type="long">
  <generator class="uuid.string"/>
 </id>

(10)foreign算法
    使用另外一个相关联的对象的标识符。通常和<one-to-one>联合起来使用。
例:
 <id name="id" column="ID" type="long">
  <generator class="foreign"/>
 </id>

(11)select算法
    通过数据库触发器选择一些唯一主键的行并返回主键盘值来分配一个主键。
    例如下例中,定义了一个命名为socialSecurityNumber的唯一值属性,它是一个自然键(natural key),命名为ID的代理键(surrogate key)的值由触发器生成。
 <id name="id" column="ID" type="long">
  <generator class="select">
   <param name="key">socialSecurityNumber</param>
  </generator>
 </id>

    由于常用的数据库,如Oracle、DB2、SQL Server、MySQL等,都提供了易用的主键生成机制(Auto-Increase字段或Sequence)。我们可以在数据库提供的主键盘生成机制上,采用native的主键盘生成方式。不过值得注意的是,一些数据库提供的主键生成机制在效率上未必最佳,大量并发insert数据时可能会引起表之间的互锁。数据库提供的主键生成机制,往往是通过在一个内部表中保存当前主键状态(如对于自增型主键而言,此内部表中主就维护着当前的最大值和递增量),之后每次插入数据会读取这个最大值,然后加上递增量作为新记录的主键,之后再把这个新的最大值勤更新回内部表中。这样,一次insert操作可能导致数据库内部多次表读写操作,同时伴随的还有数据的加锁/解锁操作,这对性能产生了较大影响。因此,对于并发insert要求较高的系统,推荐采用uuid.hex作为主键生成机制。