📄 persistent-classes.html
字号:
<html><head> <meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1"> <title>第 5 章 持久化类(Persistent Classes)</title><link rel="stylesheet" href="../shared/css/html.css" type="text/css"><meta name="generator" content="DocBook XSL Stylesheets V1.65.1"><link rel="home" href="index.html" title="HIBERNATE - 符合Java习惯的关系数据库持久化"><link rel="up" href="index.html" title="HIBERNATE - 符合Java习惯的关系数据库持久化"><link rel="previous" href="session-configuration.html" title="第 4 章 
 配置
 "><link rel="next" href="mapping.html" title="第 6 章 对象/关系数据库映射基础(Basic O/R Mapping)"></head><body bgcolor="white" text="black" link="#0000FF" vlink="#840084" alink="#0000FF"><div class="navheader"><table width="100%" summary="Navigation header"><tr><th colspan="3" align="center">第 5 章 持久化类(Persistent Classes)</th></tr><tr><td width="20%" align="left"><a accesskey="p" href="session-configuration.html">上一页</a> </td><th width="60%" align="center"> </th><td width="20%" align="right"> <a accesskey="n" href="mapping.html">下一页</a></td></tr></table><hr></div><div class="chapter" lang="zh-cn"><div class="titlepage"><div><div><h2 class="title"><a name="persistent-classes"></a>第 5 章 持久化类(Persistent Classes)</h2></div></div><div></div></div><p> 在应用程序中,用来实现业务问题实体的(如,在电子商务应用程序中的Customer和Order) 类就是持久化类。不能认为所有的持久化类的实例都是持久的状态——一个实例的状态也可能 是瞬时的或脱管的。 </p><p> 如果这些持久化类遵循一些简单的规则,Hibernate能够工作得最好,这些规则被称作, 简单传统Java对象(POJO:Plain Old Java Object)编程模型。但是这些规则没有一个是必需的。 实际上,Hibernate3对于你的持久化类几乎不做任何设想。你可以用其他的方法来表达领域模型: 比如,使用<tt class="literal">Map</tt>实例的树型结构。 </p><div class="sect1" lang="zh-cn"><div class="titlepage"><div><div><h2 class="title" style="clear: both"><a name="persistent-classes-pojo"></a>5.1. 一个简单的POJO例子</h2></div></div><div></div></div><p> 大多数Java程序需要用一个持久化类来表示猫科动物。 </p><pre class="programlisting">package eg;import java.util.Set;import java.util.Date;public class Cat { private Long id; // identifier private Date birthdate; private Color color; private char sex; private float weight; private int litterId; private Cat mother; private Set kittens = new HashSet(); private void setId(Long id) { this.id=id; } public Long getId() { return id; } void setBirthdate(Date date) { birthdate = date; } public Date getBirthdate() { return birthdate; } void setWeight(float weight) { this.weight = weight; } public float getWeight() { return weight; } public Color getColor() { return color; } void setColor(Color color) { this.color = color; } void setSex(char sex) { this.sex=sex; } public char getSex() { return sex; } void setLitterId(int id) { this.litterId = id; } public int getLitterId() { return litterId; } void setMother(Cat mother) { this.mother = mother; } public Cat getMother() { return mother; } void setKittens(Set kittens) { this.kittens = kittens; } public Set getKittens() { return kittens; } // addKitten not needed by Hibernate public void addKitten(Cat kitten) { kitten.setMother(this); kitten.setLitterId( kittens.size() ); kittens.add(kitten); }}</pre><p> 这里要遵循四条主要的规则: </p><div class="sect2" lang="zh-cn"><div class="titlepage"><div><div><h3 class="title"><a name="persistent-classes-pojo-accessors"></a>5.1.1. 为持久化字段声明访问器(accessors)和是否可变的标志(mutators)</h3></div></div><div></div></div><p> <tt class="literal">Cat</tt>为它的所有持久化字段声明了访问方法。很多其他ORM工具直接对 实例变量进行持久化。我们相信从持久化机制中分离这种实现细节要好得多。 Hibernate持久化JavaBeans风格的属性,认可如下形式的方法名: <tt class="literal">getFoo</tt>, <tt class="literal">isFoo</tt> 和 <tt class="literal">setFoo</tt>。 如果需要,你总是可以切换特定的属性的指示字段的访问方法。 </p><p> 属性<span class="emphasis"><em>不需要</em></span>要声明为public的。Hibernate默认使用 <tt class="literal">protected</tt>或<tt class="literal">private</tt>的get/set方法对, 对属性进行持久化。 </p></div><div class="sect2" lang="zh-cn"><div class="titlepage"><div><div><h3 class="title"><a name="persistent-classes-pojo-constructor"></a>5.1.2. 实现一个默认的(即无参数的)构造方法(constructor)</h3></div></div><div></div></div><p> <tt class="literal">Cat</tt>有一个无参数的构造方法。所有的持久化类都必须有一个 默认的构造方法(可以不是public的),这样的话Hibernate就可以使用 <tt class="literal">Constructor.newInstance()</tt>来实例化它们。 我们建议,在Hibernate中,为了运行期代理的生成,构造方法至少是 <span class="emphasis"><em>包(package)</em></span>内可见的。 </p></div><div class="sect2" lang="zh-cn"><div class="titlepage"><div><div><h3 class="title"><a name="persistent-classes-pojo-identifier"></a>5.1.3. 提供一个标识属性(identifier property)(可选) </h3></div></div><div></div></div><p> <tt class="literal">Cat</tt>有一个属性叫做<tt class="literal">id</tt>。这个属性映射数据库表的主 键字段。这个属性可以叫任何名字,其类型可以是任何的原始类型、原始类型的包装类型、 <tt class="literal">java.lang.String</tt> 或者是 <tt class="literal">java.util.Date</tt>。 (如果你的老式数据库表有联合主键,你甚至可以用一个用户自定义的类,该类拥有这些类型 的属性。参见后面的关于联合标识符的章节。) </p><p> 标识符属性是可选的。可以不用管它,让Hibernate内部来追踪对象的识别。 不推荐使用这个属性。 </p><p> 实际上,一些功能只对那些声明了标识符属性的类起作用: </p><div class="itemizedlist"><ul type="disc" compact><li><p> 托管对象的传播性重新(和session)关联(级联更新或级联合并) ——参阅 <a href="objectstate.html#objectstate-transitive" title="11.11. 传播性持久化(transitive persistence)">第 11.11 节 “传播性持久化(transitive persistence)”</a> </p></li><li><p> <tt class="literal">Session.saveOrUpdate()</tt> </p></li><li><p> <tt class="literal">Session.merge()</tt> </p></li></ul></div><p> 我们建议你对持久化类声明命名一致的标识属性。我们还建议你使用一 个可以为空(也就是说,不是原始类型)的类型。 </p></div><div class="sect2" lang="zh-cn"><div class="titlepage"><div><div><h3 class="title"><a name="persistent-classes-pojo-final"></a>5.1.4. 使用非final的类 (可选)</h3></div></div><div></div></div><p> <span class="emphasis"><em>代理(proxies)</em></span>是Hibernate的一个重要的功能,它依赖的条件是,持久 化类或者是非final的,或者是实现了一个所有方法都声明为public的接口。 </p><p> 你可以用Hibernate持久化一个没有实现任何接口的<tt class="literal">final</tt>类,但是你 不能使用代理来延迟关联加载,这会限制你进行性能优化的选择。 </p><p> 你也应该避免在非final类中声明 <tt class="literal">public final</tt>的方法。如果你想使用一 个有<tt class="literal">public final</tt>方法的类,你必须通过设置<tt class="literal">lazy="false"</tt> 来明确的禁用代理。 </p></div></div><div class="sect1" lang="zh-cn"><div class="titlepage"><div><div><h2 class="title" style="clear: both"><a name="persistent-classes-inheritance"></a>5.2. 实现继承(Inheritance)</h2></div></div><div></div></div><p> 子类也必须遵守第一条和第二条规则。它从超类<tt class="literal">Cat</tt>继承了标识属性。 </p><pre class="programlisting">package eg;public class DomesticCat extends Cat { private String name; public String getName() { return name; } protected void setName(String name) { this.name=name; }}</pre></div><div class="sect1" lang="zh-cn"><div class="titlepage"><div><div><h2 class="title" style="clear: both"><a name="persistent-classes-equalshashcode"></a>5.3. 实现<tt class="literal">equals()</tt>和<tt class="literal">hashCode()</tt></h2></div></div><div></div></div><p> 如果你有如下需求,你必须重载
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -