📄 persistent-classes.html
字号:
<html><head> <meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1"> <title>第 4 章 持久化类(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="第 3 章 SessionFactory配置"><link rel="next" href="mapping.html" title="第 5 章 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">第 4 章 持久化类(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>第 4 章 持久化类(Persistent Classes)</h2></div></div><div></div></div><p> 持久化类是应用程序用来解决商业问题的类(比如,在电子交易程序中的Customer和Order)。持久化类,就如同它的名字暗示的,是短暂存在的,它的实例会被持久性保存于数据库中。 </p><p> 如果这些类符合简单的规则,Hibernate能够工作得最好,这些规则就是Plain Old Java Object (POJO,简单传统Java对象)编程模型。 </p><div class="sect1" lang="zh-cn"><div class="titlepage"><div><div><h2 class="title" style="clear: both"><a name="persistent-classes-pojo"></a>4.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 String name; private Date birthdate; private Cat mate; private Set kittens private Color color; private char sex; private float weight; private void setId(Long id) { this.id=id; } public Long getId() { return id; } void setName(String name) { this.name = name; } public String getName() { return name; } void setMate(Cat mate) { this.mate = mate; } public Cat getMate() { return mate; } 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 setKittens(Set kittens) { this.kittens = kittens; } public Set getKittens() { return kittens; } // addKitten not needed by Hibernate public void addKitten(Cat kitten) { kittens.add(kitten); } void setSex(char sex) { this.sex=sex; } public char getSex() { return sex; }}</pre><p> 有四条主要的规则: </p><div class="sect2" lang="zh-cn"><div class="titlepage"><div><div><h3 class="title"><a name="persistent-classes-pojo-accessors"></a>4.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可以对default,<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>4.1.2. 实现一个默认的构造方法(constructor)</h3></div></div><div></div></div><p> <tt class="literal">Cat</tt>有一个显式的无参数默认构造方法。所有的持久化类都必须具有一个默认的构造方法(可以不是public的),这样的话Hibernate就可以使用<tt class="literal">Constructor.newInstance()</tt>来实例化它们。 </p></div><div class="sect2" lang="zh-cn"><div class="titlepage"><div><div><h3 class="title"><a name="persistent-classes-pojo-identifier"></a>4.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> 级联更新(Cascaded updates)(参阅“自管理生命周期的对象(Lifecycle Objects)”) </p></li><li><p> <tt class="literal">Session.saveOrUpdate()</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>4.1.4. 建议使用不是final的类 (可选)</h3></div></div><div></div></div><p> Hibernate的关键功能之一,<span class="emphasis"><em>代理(proxies)</em></span>,要求持久化类不是final的,或者是一个全部方法都是public的接口的具体实现。 </p><p> 你可以对一个<tt class="literal">final</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>4.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>4.3. 实现<tt class="literal">equals()</tt>和<tt class="literal">hashCode()</tt></h2></div></div><div></div></div><p> 如果你需要混合使用持久化类(比如,在一个<tt class="literal">Set</tt>中),你必须重载<tt class="literal">equals()</tt> 和 <tt class="literal">hashCode()</tt>方法。 </p><p> <span class="emphasis"><em>这仅适用于那些在两个不同的<tt class="literal">Session</tt>中装载的对象,Hibernate在单个<tt class="literal">Session</tt>中仅保证JVM 辨别(<tt class="literal"> a == b </tt>,<tt class="literal">equals()</tt>的默认实现)!</em></span> </p><p> 就算两个对象<tt class="literal">a</tt>和<tt class="literal">b</tt>实际是同一行数据库内容(它们拥有同样的主键值作为辨识符),我们也不能保证在特定的<tt class="literal">Session</tt> 之外它们是同一个Java实例。 </p><p> 最显而易见的实现<tt class="literal">equals()</tt>/<tt class="literal">hashCode()</tt>方法的办法就是比较两个对象的标识值。如果这个值是同堂的,他们必定是直线同一条数据库行,所以它们是相等的(如果都被加入到<tt class="literal">Set</tt>,在<tt class="literal">Set</tt>中只应该出现一个元素)。不幸的是,我们不能使用这种办法。Hibernate只会对已经持久化的对象赋予标识值,新创建的实例将不会有任何标识符值!我们推荐使用<span class="emphasis"><em>商业关键字相等</em></span>原则来实现<tt class="literal">equals()</tt>和<tt class="literal">hashCode()</tt>。 </p><p> 商业关键字相等意味着<tt class="literal">equals()</tt>方法只比较那些组成商业关键字的属性,它对应着真实世界中的实例(<span class="emphasis"><em>自然</em></span>的候选关键字) </p><pre class="programlisting">public class Cat { ... public boolean equals(Object other) { if (this == other) return true; if (!(other instanceof Cat)) return false; final Cat cat = (Cat) other; if (!getName().equals(cat.getName())) return false; if (!getBirthday().equals(cat.getBirthday())) return false; return true; } public int hashCode() { int result; result = getName().hashCode(); result = 29 * result + getBirthday().hashCode(); return result; }
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -