怎样对带有不可序列化属性的Java对象进行序列化

出于很多原因我们想使用自定义的序列化方法取代Java默认的机制。一个最常见的原因是提高性能,而另一个原因是有时候我们无法使用默认的序列化方法。在这篇文章中,我们具体来讨论怎样通过定制的序列化方法,对一个较大的、带有不可序列化属性的对象进行序列化。

下面这段代码定义了一个简单的类。它可以把一个给定的对象序列化到一个指定的文件,或者从相同的文件中把对象反序列化出来。在这片文章中,我将使用这个类进行演示。

SerializationDemonstrator.java

package dustin.examples.serialization;  

import static java.lang.System.out;        
import java.io.FileInputStream;  
import java.io.FileOutputStream;  
import java.io.IOException;  
import java.io.ObjectInputStream;  
import java.io.ObjectOutputStream;  

/** 
 * Simple serialization/deserialization demonstrator. 
 *  
 * @author Dustin 
 */ 
public class SerializationDemonstrator  
{  
   /** 
    * Serialize the provided object to the file of the provided name. 
    * @param objectToSerialize Object that is to be serialized to file; it is 
    *     best that this object have an individually overridden toString() 
    *     implementation as that is used by this method for writing our status. 
    * @param fileName Name of file to which object is to be serialized. 
    * @throws IllegalArgumentException Thrown if either provided parameter is null. 
    */ 
   public static <T> void serialize(final T objectToSerialize, final String fileName)  
   {  
      if (fileName == null)  
      {  
         throw new IllegalArgumentException(  
            "Name of file to which to serialize object to cannot be null.");  
      }  
      if (objectToSerialize == null)  
      {  
         throw new IllegalArgumentException("Object to be serialized cannot be null.");  
      }  
      try (FileOutputStream fos = new FileOutputStream(fileName);  
           ObjectOutputStream oos = new ObjectOutputStream(fos))  
      {  
         oos.writeObject(objectToSerialize);  
         out.println("Serialization of Object " + objectToSerialize + " completed.");  
      }  
      catch (IOException ioException)  
      {  
         ioException.printStackTrace();  
      }  
   }  

   /** 
    * Provides an object deserialized from the file indicated by the provided 
    * file name. 
    *  
    * @param <T> Type of object to be deserialized. 
    * @param fileToDeserialize Name of file from which object is to be deserialized. 
    * @param classBeingDeserialized Class definition of object to be deserialized 
    *    from the file of the provided name/path; it is recommended that this 
    *    class define its own toString() implementation as that will be used in 
    *    this method's status output. 
    * @return Object deserialized from provided filename as an instance of the 
    *    provided class; may be null if something goes wrong with deserialization. 
    * @throws IllegalArgumentException Thrown if either provided parameter is null. 
    */ 
   public static <T> T deserialize(final String fileToDeserialize, final Class<T> classBeingDeserialized)  
   {  
      if (fileToDeserialize == null)  
      {  
         throw new IllegalArgumentException("Cannot deserialize from a null filename.");  
      }  
      if (classBeingDeserialized == null)  
      {  
         throw new IllegalArgumentException("Type of class to be deserialized cannot be null.");  
      }  
      T objectOut = null;  
      try (FileInputStream fis = new FileInputStream(fileToDeserialize);  
           ObjectInputStream ois = new ObjectInputStream(fis))  
      {  
         objectOut = (T) ois.readObject();  
         out.println("Deserialization of Object " + objectOut + " is completed.");  
      }  
      catch (IOException | ClassNotFoundException exception)  
      {  
         exception.printStackTrace();  
      }  
      return objectOut;  
   }  
}  

下面这段代码给出了一个使用SerializationDemonstrator类序列化和反序列化标准的Java字符串的例子。字符串是支持序列化的。代码之后的截图显示了在Netbeans中运行该类的serialize和deserialize方法后的输出。

Running SerializationDemonstrator Methods on String

SerializationDemonstrator.serialize("Inspired by Actual Events", "string.dat");  
final String stringOut = SerializationDemonstrator.deserialize("string.dat", String.class);

这里写图片描述
下面这两段代码定义了Person和CityState两个类。CityState是Person的一个属性。可以看到尽管Person实现了Serializable接口,CityState却没有。

Person.java

package dustin.examples.serialization;  

import java.io.Serializable;  

/** 
 * Person class. 
 *  
 * @author Dustin 
 */ 
public class Person implements Serializable  
{  
   private String lastName;  
   private String firstName;  
   private CityState cityAndState;  

   public Person(  
      final String newLastName, final String newFirstName,  
      final CityState newCityAndState)  
   {  
      this.lastName = newLastName;  
      this.firstName = newFirstName;  
      this.cityAndState = newCityAndState;  
   }  

   public String getFirstName()  
   {  
      return this.firstName;  
   }  

   public String getLastName()  
   {  
      return this.lastName;  
   }  

   @Override 
   public String toString()  
   {  
      return this.firstName + " " + this.lastName + " of " + this.cityAndState;  
   }  
}  

CityState.java

package dustin.examples.serialization;  

/** 
 * Simple class storing city and state names that is NOT Serializable. 
 *  
 * @author Dustin 
 */ 
public class CityState  
{  
   private final String cityName;  
   private final String stateName;  

   public CityState(final String newCityName, final String newStateName)  
   {  
      this.cityName = newCityName;  
      this.stateName = newStateName;  
   }  

   public String getCityName()  
   {  
      return this.cityName;  
   }  

   public String getStateName()  
   {  
      return this.stateName;  
   }  

   @Override 
   public String toString()  
   {  
      return this.cityName + ", " + this.stateName;  
   }  
}  

下面这段代码演示了使用SerializationDemonstrator序列化Person类。由于包含了一个不可序列化的属性CityState,在之后截图里,我们可以看到Netbean抛出了异常。

Running SerializationDemonstrator Methods on Serializable Person with Non-Serializable CityState

final Person personIn = new Person("Flintstone", "Fred", new CityState("Bedrock", "Cobblestone"));  
SerializationDemonstrator.serialize(personIn, "person.dat");  

final Person personOut = SerializationDemonstrator.deserialize("person.dat", Person.class);  

这里写图片描述

在这个例子里,由于CityState类是我们自己写的,我们可以使它支持序列化。但是如果这个类属于一个第三方的框架或者库,我们就很难去修改这个类。但是我们可以修改Person类,通过使用自定义的序列化和反序列化方法,使它和CityState类一起正常工作。下面这段代码定义了一个从Person类改过来的SerializablePerson类。

SerializablePerson.java

package dustin.examples.serialization;  

import java.io.IOException;  
import java.io.InvalidObjectException;  
import java.io.ObjectInputStream;  
import java.io.ObjectOutputStream;  
import java.io.ObjectStreamException;  
import java.io.Serializable;  

/** 
 * Person class. 
 *  
 * @author Dustin 
 */ 
public class SerializablePerson implements Serializable  
{  
   private String lastName;  
   private String firstName;  
   private CityState cityAndState;  

   public SerializablePerson(  
      final String newLastName, final String newFirstName,  
      final CityState newCityAndState)  
   {  
      this.lastName = newLastName;  
      this.firstName = newFirstName;  
      this.cityAndState = newCityAndState;  
   }  

   public String getFirstName()  
   {  
      return this.firstName;  
   }  

   public String getLastName()  
   {  
      return this.lastName;  
   }  

   @Override 
   public String toString()  
   {  
      return this.firstName + " " + this.lastName + " of " + this.cityAndState;  
   }  

   /** 
    * Serialize this instance. 
    *  
    * @param out Target to which this instance is written. 
    * @throws IOException Thrown if exception occurs during serialization. 
    */ 
   private void writeObject(final ObjectOutputStream out) throws IOException  
   {  
      out.writeUTF(this.lastName);  
      out.writeUTF(this.firstName);  
      out.writeUTF(this.cityAndState.getCityName());  
      out.writeUTF(this.cityAndState.getStateName());  
   }  

   /** 
    * Deserialize this instance from input stream. 
    *  
    * @param in Input Stream from which this instance is to be deserialized. 
    * @throws IOException Thrown if error occurs in deserialization. 
    * @throws ClassNotFoundException Thrown if expected class is not found. 
    */ 
   private void readObject(final ObjectInputStream in) throws IOException, ClassNotFoundException  
   {  
      this.lastName = in.readUTF();  
      this.firstName = in.readUTF();  
      this.cityAndState = new CityState(in.readUTF(), in.readUTF());  
   }  

   private void readObjectNoData() throws ObjectStreamException  
   {  
      throw new InvalidObjectException("Stream data required");  
   }  
}  

在上面这段代码中,SerializablePerson有自定义的writeobject和readObject方法。它们以适当的方式处理CityState的序列化和反序列化。下面这段代码使用SerializationDemonstrator运行了这个类,我们可以看到这次的运行是成功的。

Running SerializationDemonstrator on SerializablePerson

final SerializablePerson personIn = new SerializablePerson("Flintstone", "Fred", new CityState("Bedrock", "Cobblestone"));  
SerializationDemonstrator.serialize(personIn, "person1.dat");  

final SerializablePerson personOut = SerializationDemonstrator.deserialize("person1.dat", SerializablePerson.class);  

这里写图片描述

上面描述的这个方法可以允许我们在一个可序列化的类中使用不可序列化的属性,而且不需要transient。现在看上去已经挺不错了,但是如果前面这个CityState要在多个需要序列化的类中使用,更好的方式是用一个支持序列化的Decorator去修饰CityState。然后在那些需要做序列化的类中使用这个Decorator。下面这段代码定义了SerializableCityState。它是CityState的一个支持序列化的Decorator版本。

SerializableCityState.java

package dustin.examples.serialization;  

import java.io.IOException;  
import java.io.InvalidObjectException;  
import java.io.ObjectInputStream;  
import java.io.ObjectOutputStream;  
import java.io.ObjectStreamException;  
import java.io.Serializable;  

/** 
 * Simple class storing city and state names that IS Serializable. This class 
 * decorates the non-Serializable CityState class and adds Serializability. 
 *  
 * @author Dustin 
 */ 
public class SerializableCityState implements Serializable  
{  
   private CityState cityState;  

   public SerializableCityState(final String newCityName, final String newStateName)  
   {  
      this.cityState = new CityState(newCityName, newStateName);  
   }  

   public String getCityName()  
   {  
      return this.cityState.getCityName();  
   }  

   public String getStateName()  
   {  
      return this.cityState.getStateName();  
   }  

   @Override 
   public String toString()  
   {  
      return this.cityState.toString();  
   }  

   /** 
    * Serialize this instance. 
    *  
    * @param out Target to which this instance is written. 
    * @throws IOException Thrown if exception occurs during serialization. 
    */ 
   private void writeObject(final ObjectOutputStream out) throws IOException  
   {  
      out.writeUTF(this.cityState.getCityName());  
      out.writeUTF(this.cityState.getStateName());  
   }  

   /** 
    * Deserialize this instance from input stream. 
    *  
    * @param in Input Stream from which this instance is to be deserialized. 
    * @throws IOException Thrown if error occurs in deserialization. 
    * @throws ClassNotFoundException Thrown if expected class is not found. 
    */ 
   private void readObject(final ObjectInputStream in) throws IOException, ClassNotFoundException  
   {  
      this.cityState = new CityState(in.readUTF(), in.readUTF());  
   }  

   private void readObjectNoData() throws ObjectStreamException  
   {  
      throw new InvalidObjectException("Stream data required");  
   }  
}  

这个可序列化的Decorator可以在Person类中直接使用。由于所有的属性都支持序列化,Person类可以使用默认的序列化方法。下面这段代码定义了一个从Person类改过来的Person2类。

Person2.java

package dustin.examples.serialization;  

import java.io.Serializable;  

/** 
 * Person class. 
 *  
 * @author Dustin 
 */ 
public class Person2 implements Serializable  
{  
   private final String lastName;  
   private final String firstName;  
   private final SerializableCityState cityAndState;  

   public Person2(  
      final String newLastName, final String newFirstName,  
      final SerializableCityState newCityAndState)  
   {  
      this.lastName = newLastName;  
      this.firstName = newFirstName;  
      this.cityAndState = newCityAndState;  
   }  

   public String getFirstName()  
   {  
      return this.firstName;  
   }  

   public String getLastName()  
   {  
      return this.lastName;  
   }  

   @Override 
   public String toString()  
   {  
      return this.firstName + " " + this.lastName + " of " + this.cityAndState;  
   }  
}  

下面这段代码运行了这个类。之后是NetBeans输出的截图。

Running SerializationDemonstrator Against Person2/SerializableCityState

final Person2 personIn = new Person2("Flintstone", "Fred", new SerializableCityState("Bedrock", "Cobblestone"));  
SerializationDemonstrator.serialize(personIn, "person2.dat");  

final Person2 personOut = SerializationDemonstrator.deserialize("person2.dat", Person2.class);  

这里写图片描述

通过使用定制的序列化方法,可以在不使用transient的情况下,对一个带有不可序列化属性的类进行序列化。当你要在一个需要序列化的类中使用不可序列化的类型,并且这些类型不能被修改时,这是一个有用的技术。

原文链接: marxsoftware 翻译: ImportNew.com - keywords
译文链接: http://www.importnew.com/10705.html

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
第1章 Java应用分层架构及软件模型  1.1 应用程序的分层体系结构   1.1.1 区分物理层和逻辑层   1.1.2 软件层的特征   1.1.3 软件分层的优点   1.1.4 软件分层的缺点   1.1.5 Java应用的持久化层  1.2 软件的模型   1.2.1 概念模型   1.2.2 关系数据模型   1.2.3 域模型   1.2.4 域对象   1.2.5 域对象之间的关系   1.2.6 域对象的持久化概念  1.3 小结  1.4 思考题 第2章 Java对象持久化技术概述  2.1 直接通过JDBC API来持久化实体域对象  2.2 ORM简介   2.2.1 对象-关系映射的概念   2.2.2 ORM中间件的基本使用方法   2.2.3 常用的ORM中间件  2.3 实体域对象的其他持久化模式   2.3.1 主动域对象模式   2.3.2 JDO模式   2.3.3 CMP模式  2.4 Hibernate API简介   2.4.1 Hibernate的核心接口   2.4.2 事件处理接口   2.4.3 Hibernate映射型接口   2.4.4 可供扩展的接口  2.5 小结  2.6 思考题 第3章 第一个Hibernate应用  3.1 创建Hibernate的配置文件  3.2 创建持久化  3.3 创建数据库Schema  3.4 创建对象-关系映射文件   3.4.1 映射文件的文档型定义(DTD)   3.4.2 把Customer持久化映射到CUSTOMERS表  3.5 通过Hibernate API操纵数据库   3.5.1 Hibernate的初始化   3.5.2 访问Hibernate的Session接口  3.6 运行helloapp应用   3.6.1 创建运行本书范例的系统环境   3.6.2 创建helloapp应用的目录结构   3.6.3 把helloapp应用作为独立应用程序运行   3.6.4 把helloapp应用作为Java Web应用运行  3.7 小结  3.8 思考题 第4章 hbm2java和hbm2ddl工具  4.1 创建对象-关系映射文件   4.1.1 定制持久化   4.1.2 定制数据库表  4.2 建立项目的目录结构  4.3 运行hbm2java工具  4.4 运行hbm2ddl工具  4.5 使用XML格式的配置文件  4.6 小结  4.7 思考题 第5章 对象-关系映射基础  5.1 持久化属性及访问方法   5.1.1 基本属性和包装属性   5.1.2 Hibernate访问持久化属性的策略   5.1.3 在持久化的访问方法中加入程序逻辑   5.1.4 设置派生属性   5.1.5 控制insert和update语句  5.2 处理SQL引用标识符  5.3 创建命名策略  5.4 设置数据库Schema  5.5 设置的包名  5.6 运行本章的范例程序  5.7 小结  5.8 思考题 第6章 映射对象标识符  6.1 关系数据库按主键区分不同的记录   6.1.1 把主键定义为自动增长标识符型   6.1.2 从序列(Sequence)中获取自动增长的标识符 6.2 Java语言按内存地址区分不同的对象 6.3 Hibernate用对象标识符(OID)来区分对象 6.4 Hibernate的内置标识符生成器的用法   6.4.1 increment标识符生成器   6.4.2 identity标识符生成器   6.4.3 sequence标识符生成器   6.4.4 hilo标识符生成器   6.4.5 native标识符生成器  6.5 映射自然主键   6.5.1 映射单个自然主键   6.5.2 映射复合自然主键  6.6 小结  6.7 思考题 第7章 映射一对多关联关系  7.1 建立多对一的单向关联关系   7.1.1 元素的not-null属性   7.1.2 级联保存和更新  7.2 映射一对多双向关联关系   7.2.1 元素的inverse属性   7.2.2 级联删除   7.2.3 父子关系  7.3 映射一对多双向自身关联关系  7.4 改进持久化  7.5 小结  7.6 思考题 第8章 通过Hibernate操纵对象(上)  8.1 Java对象在JVM中的生命周期  8.2 理解Session的缓存   8.2.1 Session的缓存的作用   8.2.2 脏检查及清理缓存的机制  8.3 Java对象在Hibernate持久化层的状态   8.3.1 临时对象的特征   8.3.2 持久化对象的特征   8.3.3 被删除对象的特征   8.3.4 游离对象的特征  8.4 Session接口的详细用法   8.4.1 Session的save()和persist()方法   8.4.2 Session的load()和get()方法   8.4.3 Session的update()方法   8.4.4 Session的saveOrUpdate()方法   8.4.5 Session的merge()方法   8.4.6 Session的delete()方法   8.4.7 Session的replicate()方法  8.5 级联操纵对象图   8.5.1 级联保存临时对象   8.5.2 更新持久化对象   8.5.3 持久化临时对象   8.5.4 更新游离对象   8.5.5 遍历对象图  8.6 小结  8.7 思考题 第9章 通过Hibernate操纵对象(下)  9.1 与触发器协同工作  9.2 利用拦截器(Interceptor)生成审计日志  9.3 Hibernate的事件处理机制  9.4 批量处理数据   9.4.1 通过Session来进行批量操作   9.4.2 通过StatelessSession来进行批量操作   9.4.3 通过HQL来进行批量操作   9.4.4 直接通过JDBC API来进行批量操作  9.5 使用元数据  9.6 通过Hibernate调用存储过程  9.7 小结  9.8 思考题 第10章 映射组成关系  10.1 建立精粒度对象模型  10.2 建立粗粒度关系数据模型  10.3 映射组成关系   10.3.1 区分值(Value)型和实体(Entity)型   10.3.2 在应用程序中访问具有组成关系的持久化  10.4 映射复合组成关系  10.5 小结  10.6 思考题 第11章 Hibernate的映射型  11.1 Hibernate的内置映射型   11.1.1 Java基本型的Hibernate映射型   11.1.2 Java时间和日期型的Hibernate映射型   11.1.3 Java对象型的Hibernate映射型   11.1.4 JDK自带的个别Java的Hibernate映射型   11.1.5 使用Hibernate内置映射型  11.2 客户化映射型   11.2.1 用客户化映射型取代Hibernate组件   11.2.2 用UserType映射枚举型   11.2.3 实现CompositeUserType接口   11.2.4 运行本节范例程序  11.3 操纵Blob和Clob型数据  11.4 小结  11.5 思考题 第12章 映射继承关系  12.1 继承关系树的每个具体对应一个表   12.1.1 创建映射文件   12.1.2 操纵持久化对象  12.2 继承关系树的根对应一个表   12.2.1 创建映射文件   12.2.2 操纵持久化对象  12.3 继承关系树的每个对应一个表   12.3.1 创建映射文件   12.3.2 操纵持久化对象  12.4 选择继承关系的映射方式  12.5 映射多对一多态关联  12.6 小结  12.7 思考题 第13章 Java集合  13.1 Set(集)   13.1.1 Set的一般用法   13.1.2 HashSet   13.1.3 TreeSet   13.1.4 向Set中加入持久化对象  13.2 List(列表)  13.3 Map(映射)  13.4 小结  13.5 思考题 第14章 映射值型集合  14.1 映射Set(集)  14.2 映射Bag(包)  14.3 映射List(列表)  14.4 映射Map  14.5 对集合排序   14.5.1 在数据库中对集合排序   14.5.2 在内存中对集合排序  14.6 映射组件型集合  14.7 小结  14.8 思考题 第15章 映射实体关联关系  15.1 映射一对一关联   15.1.1 按照外键映射     15.1.2 按照主键映射  15.2 映射单向多对多关联  15.3 映射双向多对多关联关系   15.3.1 关联两端使用元素   15.3.2 在inverse端使用元素   15.3.3 使用组件集合   15.3.4 把多对多关联分解为两个一对多关联  15.4 小结  15.5 思考题 第16章 Hibernate的检索策略  16.1 Hibernate的检索策略简介  16.2 级别的检索策略   16.2.1 立即检索   16.2.2 延迟检索  16.3 一对多和多对多关联的检索策略   16.3.1 立即检索(lazy属性为“false”)   16.3.2 延迟检索(lazy属性为默认值“true”)   16.3.3 增强延迟检索(lazy属性为“extra”)   16.3.4 批量延迟检索和批量立即检索(使用batch-size属性)   16.3.5 用带子查询的select语句整批量初始化orders集合(fetch属性为“subselect”)   16.3.6 迫切左外连接检索(fetch属性为“join”)  16.4 多对一和一对一关联的检索策略   16.4.1 迫切左外连接检索(fetch属性为“join”)   16.4.2 延迟检索(lazy属性为默认值“proxy”)   16.4.3 无代理延迟检索(lazy属性为“no-proxy”)   16.4.4 立即检索(lazy属性为“false”)   16.4.5 批量延迟检索和批量立即检索(使用batch-size属性)  16.5 控制迫切左外连接检索的深度  16.6 在应用程序中显式指定迫切左外连接检索策略  16.7 属性级别的检索策略  16.8 小结  16.9 思考题 第17章 Hibernate的检索方式(上)  17.1 Hibernate的检索方式简介   17.1.1 HQL检索方式   17.1.2 QBC检索方式   17.1.3 本地SQL检索方式   17.1.4 关于本章范例程序   17.1.5 使用别名   17.1.6 多态查询   17.1.7 对查询结果排序   17.1.8 分页查询   17.1.9 检索单个对象(uniqueResult()方法)   17.1.10 按主键逐个处理查询结果(iterate()方法)   17.1.11 可滚动的结果集   17.1.12 在HQL查询语句中绑定参数   17.1.13 设置查询附属事项   17.1.14 在映射文件中定义命名查询语句   17.1.15 在HQL查询语句中调用函数  17.2 设定查询条件   17.2.1 比较运算   17.2.2 范围运算   17.2.3 字符串模式匹配   17.2.4 逻辑运算   17.2.5 集合运算  17.3 小结  17.4 思考题 第18章 Hibernate的检索方式(下)  18.1 连接查询   18.1.1 默认情况下关联级别的运行时检索策略   18.1.2 迫切左外连接   18.1.3 左外连接   18.1.4 内连接   18.1.5 迫切内连接   18.1.6 隐式内连接   18.1.7 右外连接   18.1.8 使用SQL风格的交叉连接和隐式内连接   18.1.9 关联级别运行时的检索策略  18.2 投影查询  18.3 报表查询   18.3.1 使用聚集函数   18.3.2 分组查询   18.3.3 优化报表查询的性能  18.4 高级查询技巧   18.4.1 动态查询   18.4.2 集合过滤   18.4.3 子查询   18.4.4 本地SQL查询   18.4.5 查询结果转换器  18.5 查询性能优化   18.5.1 iterate()方法   18.5.2 查询缓存  18.6 小结  18.7 思考题 第19章 Hibernate高级配置  19.1 配置数据库连接池   19.1.1 使用默认的数据库连接池   19.1.2 使用配置文件指定的数据库连接池   19.1.3 从容器中获得数据源   19.1.4 由Java应用本身提供数据库连接  19.2 配置事务型  19.3 把SessionFactory与JNDI绑定  19.4 配置日志  19.5 使用XML格式的配置文件  19.6 小结  19.7 思考题 第20章 声明数据库事务  20.1 数据库事务的概念  20.2 声明事务边界的方式  20.3 在mysql.exe程序中声明事务  20.4 Java应用通过JDBC API声明JDBC事务  20.5 Java应用通过Hibernate API声明JDBC事务   20.5.1 处理异常   20.5.2 Session与事务的关系   20.5.3 设定事务超时  20.6 Java应用通过Hibernate API声明JTA事务  20.7 Java应用通过JTA API声明JTA事务  20.8 小结  20.9 思考题 第21章 处理并发问题  21.1 多个事务并发运行时的并发问题   21.1.1 第一丢失更新   21.1.2 脏读   21.1.3 虚读   21.1.4 不可重复读   21.1.5 第二丢失更新  21.2 数据库系统的锁的基本原理   21.2.1 锁的多粒度性及自动锁升级   21.2.2 锁的型和兼容性   21.2.3 死锁及其防止办法  21.3 数据库的事务隔离级别   21.3.1 在mysql.exe程序中设置隔离级别   21.3.2 在应用程序中设置隔离级别  21.4 在应用程序中采用悲观锁   21.4.1 利用数据库系统的独占锁来实现悲观锁   21.4.2 由应用程序实现悲观锁  21.5 利用Hibernate的版本控制来实现乐观锁   21.5.1 使用元素   21.5.2 使用元素   21.5.3 对游离对象进行版本检查   21.5.4 强制更新版本  21.6 实现乐观锁的其他方法  21.7 小结  21.8 思考题 第22章 管理Hibernate的缓存  22.1 缓存的基本原理   22.1.1 持久化层的缓存的范围   22.1.2 持久化层的缓存的并发访问策略  22.2 Hibernate的二级缓存结构  22.3 管理Hibernate的第一级缓存  22.4 管理Hibernate的第二级缓存   22.4.1 配置进程范围内的第二级缓存   22.4.2 配置集群范围内的第二级缓存   22.4.3 在应用程序中管理第二级缓存   22.4.4 Session与第二级缓存的交互模式  22.5 运行本章的范例程序  22.6 小结  22.7 思考题 第23章 管理Session和实现对话  23.1 管理Session对象的生命周期   23.1.1 Session对象的生命周期与本地线程绑定   23.1.2 Session对象的生命周期与JTA事务绑定  23.2 实现对话   23.2.1 使用游离对象   23.2.2 使用手工清理缓存模式下的Session  23.3 小结  23.4 思考题 第24章 Hibernate与Struts框架  24.1 实现业务数据  24.2 实现业务逻辑  24.3 netstore应用的订单业务  24.4 小结 第25章 Hibernate与EJB组件  25.1 创建EJB组件   25.1.1 编写Remote接口   25.1.2 编写Home接口   25.1.3 编写Enterprise Java Bean  25.2 在业务代理中访问EJB组件  25.3 发布J2EE应用   25.3.1 在JBoss上部署EJB组件   25.3.2 在JBoss上部署Web应用   25.3.3 在JBoss上部署J2EE应用  25.4 小结 附录A 标准SQL语言的用法  A.1 数据完整性   A.1.1 实体完整性   A.1.2 域完整性   A.1.3 参照完整性  A.2 DDL数据定义语言  A.3 DML数据操纵语言  A.4 DQL数据查询语言   A.4.1 简单查询   A.4.2 连接查询   A.4.3 子查询   A.4.4 联合查询   A.4.5 报表查询 附录B Java语言的反射机制  B.1 Java Reflection API简介  B.2 运用反射机制来持久化Java对象 附录C 用XDoclet工具生成映射文件  C.1 创建带有@hibernate标记的Java源文件  C.2 建立项目的目录结构  C.3 运行XDoclet工具 附录D 发布和运行netstore应用  D.1 运行netstore所需的软件  D.2 netstore应用的目录结构  D.3 安装SAMPLEDB数据库  D.4 安装和配置JBoss服务器  D.5 发布netstore应用   D.5.1 在工作模式1下发布netstore应用   D.5.2 在工作模式2下发布netstore应用  D.6 运行netstore应用 附录E Hibernate 3升级指南  E.1 Hibernate API 变化   E.1.1 包名   E.1.2 org.hibernate.classic包   E.1.3 Hibernate所依赖的第三方软件包   E.1.4 异常模型   E.1.5 Session接口   E.1.6 createSQLQuery()   E.1.7 Lifecycle 和 Validatable 接口   E.1.8 Interceptor接口   E.1.9 UserType和CompositeUserType接口   E.1.10 FetchMode   E.1.11 PersistentEnum   E.1.12 对Blob 和Clob的支持   E.1.13 Hibernate中供扩展的API的变化  E.2 元数据的变化   E.2.1 检索策略   E.2.2 对象标识符的映射   E.2.3 集合映射   E.2.4 DTD  E.3 查询语句的变化  E.4 把Hibernate 2应用升级到Hibernate 3应用
第1章 Java应用分层架构及软件模型  1.1 应用程序的分层体系结构   1.1.1 区分物理层和逻辑层   1.1.2 软件层的特征   1.1.3 软件分层的优点   1.1.4 软件分层的缺点   1.1.5 Java应用的持久化层  1.2 软件的模型   1.2.1 概念模型   1.2.2 关系数据模型   1.2.3 域模型   1.2.4 域对象   1.2.5 域对象之间的关系   1.2.6 域对象的持久化概念  1.3 小结  1.4 思考题 第2章 Java对象持久化技术概述  2.1 直接通过JDBC API来持久化实体域对象  2.2 ORM简介   2.2.1 对象-关系映射的概念   2.2.2 ORM中间件的基本使用方法   2.2.3 常用的ORM中间件  2.3 实体域对象的其他持久化模式   2.3.1 主动域对象模式   2.3.2 JDO模式   2.3.3 CMP模式  2.4 Hibernate API简介   2.4.1 Hibernate的核心接口   2.4.2 事件处理接口   2.4.3 Hibernate映射型接口   2.4.4 可供扩展的接口  2.5 小结  2.6 思考题 第3章 第一个Hibernate应用  3.1 创建Hibernate的配置文件  3.2 创建持久化  3.3 创建数据库Schema  3.4 创建对象-关系映射文件   3.4.1 映射文件的文档型定义(DTD)   3.4.2 把Customer持久化映射到CUSTOMERS表  3.5 通过Hibernate API操纵数据库   3.5.1 Hibernate的初始化   3.5.2 访问Hibernate的Session接口  3.6 运行helloapp应用   3.6.1 创建运行本书范例的系统环境   3.6.2 创建helloapp应用的目录结构   3.6.3 把helloapp应用作为独立应用程序运行   3.6.4 把helloapp应用作为Java Web应用运行  3.7 小结  3.8 思考题 第4章 hbm2java和hbm2ddl工具  4.1 创建对象-关系映射文件   4.1.1 定制持久化   4.1.2 定制数据库表  4.2 建立项目的目录结构  4.3 运行hbm2java工具  4.4 运行hbm2ddl工具  4.5 使用XML格式的配置文件  4.6 小结  4.7 思考题 第5章 对象-关系映射基础  5.1 持久化属性及访问方法   5.1.1 基本属性和包装属性   5.1.2 Hibernate访问持久化属性的策略   5.1.3 在持久化的访问方法中加入程序逻辑   5.1.4 设置派生属性   5.1.5 控制insert和update语句  5.2 处理SQL引用标识符  5.3 创建命名策略  5.4 设置数据库Schema  5.5 设置的包名  5.6 运行本章的范例程序  5.7 小结  5.8 思考题 第6章 映射对象标识符  6.1 关系数据库按主键区分不同的记录   6.1.1 把主键定义为自动增长标识符型   6.1.2 从序列(Sequence)中获取自动增长的标识符 6.2 Java语言按内存地址区分不同的对象 6.3 Hibernate用对象标识符(OID)来区分对象 6.4 Hibernate的内置标识符生成器的用法   6.4.1 increment标识符生成器   6.4.2 identity标识符生成器   6.4.3 sequence标识符生成器   6.4.4 hilo标识符生成器   6.4.5 native标识符生成器  6.5 映射自然主键   6.5.1 映射单个自然主键   6.5.2 映射复合自然主键  6.6 小结  6.7 思考题 第7章 映射一对多关联关系  7.1 建立多对一的单向关联关系   7.1.1 元素的not-null属性   7.1.2 级联保存和更新  7.2 映射一对多双向关联关系   7.2.1 元素的inverse属性   7.2.2 级联删除   7.2.3 父子关系  7.3 映射一对多双向自身关联关系  7.4 改进持久化  7.5 小结  7.6 思考题 第8章 通过Hibernate操纵对象(上)  8.1 Java对象在JVM中的生命周期  8.2 理解Session的缓存   8.2.1 Session的缓存的作用   8.2.2 脏检查及清理缓存的机制  8.3 Java对象在Hibernate持久化层的状态   8.3.1 临时对象的特征   8.3.2 持久化对象的特征   8.3.3 被删除对象的特征   8.3.4 游离对象的特征  8.4 Session接口的详细用法   8.4.1 Session的save()和persist()方法   8.4.2 Session的load()和get()方法   8.4.3 Session的update()方法   8.4.4 Session的saveOrUpdate()方法   8.4.5 Session的merge()方法   8.4.6 Session的delete()方法   8.4.7 Session的replicate()方法  8.5 级联操纵对象图   8.5.1 级联保存临时对象   8.5.2 更新持久化对象   8.5.3 持久化临时对象   8.5.4 更新游离对象   8.5.5 遍历对象图  8.6 小结  8.7 思考题 第9章 通过Hibernate操纵对象(下)  9.1 与触发器协同工作  9.2 利用拦截器(Interceptor)生成审计日志  9.3 Hibernate的事件处理机制  9.4 批量处理数据   9.4.1 通过Session来进行批量操作   9.4.2 通过StatelessSession来进行批量操作   9.4.3 通过HQL来进行批量操作   9.4.4 直接通过JDBC API来进行批量操作  9.5 使用元数据  9.6 通过Hibernate调用存储过程  9.7 小结  9.8 思考题 第10章 映射组成关系  10.1 建立精粒度对象模型  10.2 建立粗粒度关系数据模型  10.3 映射组成关系   10.3.1 区分值(Value)型和实体(Entity)型   10.3.2 在应用程序中访问具有组成关系的持久化  10.4 映射复合组成关系  10.5 小结  10.6 思考题 第11章 Hibernate的映射型  11.1 Hibernate的内置映射型   11.1.1 Java基本型的Hibernate映射型   11.1.2 Java时间和日期型的Hibernate映射型   11.1.3 Java对象型的Hibernate映射型   11.1.4 JDK自带的个别Java的Hibernate映射型   11.1.5 使用Hibernate内置映射型  11.2 客户化映射型   11.2.1 用客户化映射型取代Hibernate组件   11.2.2 用UserType映射枚举型   11.2.3 实现CompositeUserType接口   11.2.4 运行本节范例程序  11.3 操纵Blob和Clob型数据  11.4 小结  11.5 思考题 第12章 映射继承关系  12.1 继承关系树的每个具体对应一个表   12.1.1 创建映射文件   12.1.2 操纵持久化对象  12.2 继承关系树的根对应一个表   12.2.1 创建映射文件   12.2.2 操纵持久化对象  12.3 继承关系树的每个对应一个表   12.3.1 创建映射文件   12.3.2 操纵持久化对象  12.4 选择继承关系的映射方式  12.5 映射多对一多态关联  12.6 小结  12.7 思考题 第13章 Java集合  13.1 Set(集)   13.1.1 Set的一般用法   13.1.2 HashSet   13.1.3 TreeSet   13.1.4 向Set中加入持久化对象  13.2 List(列表)  13.3 Map(映射)  13.4 小结  13.5 思考题 第14章 映射值型集合  14.1 映射Set(集)  14.2 映射Bag(包)  14.3 映射List(列表)  14.4 映射Map  14.5 对集合排序   14.5.1 在数据库中对集合排序   14.5.2 在内存中对集合排序  14.6 映射组件型集合  14.7 小结  14.8 思考题 第15章 映射实体关联关系  15.1 映射一对一关联   15.1.1 按照外键映射     15.1.2 按照主键映射  15.2 映射单向多对多关联  15.3 映射双向多对多关联关系   15.3.1 关联两端使用元素   15.3.2 在inverse端使用元素   15.3.3 使用组件集合   15.3.4 把多对多关联分解为两个一对多关联  15.4 小结  15.5 思考题 第16章 Hibernate的检索策略  16.1 Hibernate的检索策略简介  16.2 级别的检索策略   16.2.1 立即检索   16.2.2 延迟检索  16.3 一对多和多对多关联的检索策略   16.3.1 立即检索(lazy属性为“false”)   16.3.2 延迟检索(lazy属性为默认值“true”)   16.3.3 增强延迟检索(lazy属性为“extra”)   16.3.4 批量延迟检索和批量立即检索(使用batch-size属性)   16.3.5 用带子查询的select语句整批量初始化orders集合(fetch属性为“subselect”)   16.3.6 迫切左外连接检索(fetch属性为“join”)  16.4 多对一和一对一关联的检索策略   16.4.1 迫切左外连接检索(fetch属性为“join”)   16.4.2 延迟检索(lazy属性为默认值“proxy”)   16.4.3 无代理延迟检索(lazy属性为“no-proxy”)   16.4.4 立即检索(lazy属性为“false”)   16.4.5 批量延迟检索和批量立即检索(使用batch-size属性)  16.5 控制迫切左外连接检索的深度  16.6 在应用程序中显式指定迫切左外连接检索策略  16.7 属性级别的检索策略  16.8 小结  16.9 思考题 第17章 Hibernate的检索方式(上)  17.1 Hibernate的检索方式简介   17.1.1 HQL检索方式   17.1.2 QBC检索方式   17.1.3 本地SQL检索方式   17.1.4 关于本章范例程序   17.1.5 使用别名   17.1.6 多态查询   17.1.7 对查询结果排序   17.1.8 分页查询   17.1.9 检索单个对象(uniqueResult()方法)   17.1.10 按主键逐个处理查询结果(iterate()方法)   17.1.11 可滚动的结果集   17.1.12 在HQL查询语句中绑定参数   17.1.13 设置查询附属事项   17.1.14 在映射文件中定义命名查询语句   17.1.15 在HQL查询语句中调用函数  17.2 设定查询条件   17.2.1 比较运算   17.2.2 范围运算   17.2.3 字符串模式匹配   17.2.4 逻辑运算   17.2.5 集合运算  17.3 小结  17.4 思考题 第18章 Hibernate的检索方式(下)  18.1 连接查询   18.1.1 默认情况下关联级别的运行时检索策略   18.1.2 迫切左外连接   18.1.3 左外连接   18.1.4 内连接   18.1.5 迫切内连接   18.1.6 隐式内连接   18.1.7 右外连接   18.1.8 使用SQL风格的交叉连接和隐式内连接   18.1.9 关联级别运行时的检索策略  18.2 投影查询  18.3 报表查询   18.3.1 使用聚集函数   18.3.2 分组查询   18.3.3 优化报表查询的性能  18.4 高级查询技巧   18.4.1 动态查询   18.4.2 集合过滤   18.4.3 子查询   18.4.4 本地SQL查询   18.4.5 查询结果转换器  18.5 查询性能优化   18.5.1 iterate()方法   18.5.2 查询缓存  18.6 小结  18.7 思考题 第19章 Hibernate高级配置  19.1 配置数据库连接池   19.1.1 使用默认的数据库连接池   19.1.2 使用配置文件指定的数据库连接池   19.1.3 从容器中获得数据源   19.1.4 由Java应用本身提供数据库连接  19.2 配置事务型  19.3 把SessionFactory与JNDI绑定  19.4 配置日志  19.5 使用XML格式的配置文件  19.6 小结  19.7 思考题 第20章 声明数据库事务  20.1 数据库事务的概念  20.2 声明事务边界的方式  20.3 在mysql.exe程序中声明事务  20.4 Java应用通过JDBC API声明JDBC事务  20.5 Java应用通过Hibernate API声明JDBC事务   20.5.1 处理异常   20.5.2 Session与事务的关系   20.5.3 设定事务超时  20.6 Java应用通过Hibernate API声明JTA事务  20.7 Java应用通过JTA API声明JTA事务  20.8 小结  20.9 思考题 第21章 处理并发问题  21.1 多个事务并发运行时的并发问题   21.1.1 第一丢失更新   21.1.2 脏读   21.1.3 虚读   21.1.4 不可重复读   21.1.5 第二丢失更新  21.2 数据库系统的锁的基本原理   21.2.1 锁的多粒度性及自动锁升级   21.2.2 锁的型和兼容性   21.2.3 死锁及其防止办法  21.3 数据库的事务隔离级别   21.3.1 在mysql.exe程序中设置隔离级别   21.3.2 在应用程序中设置隔离级别  21.4 在应用程序中采用悲观锁   21.4.1 利用数据库系统的独占锁来实现悲观锁   21.4.2 由应用程序实现悲观锁  21.5 利用Hibernate的版本控制来实现乐观锁   21.5.1 使用元素   21.5.2 使用元素   21.5.3 对游离对象进行版本检查   21.5.4 强制更新版本  21.6 实现乐观锁的其他方法  21.7 小结  21.8 思考题 第22章 管理Hibernate的缓存  22.1 缓存的基本原理   22.1.1 持久化层的缓存的范围   22.1.2 持久化层的缓存的并发访问策略  22.2 Hibernate的二级缓存结构  22.3 管理Hibernate的第一级缓存  22.4 管理Hibernate的第二级缓存   22.4.1 配置进程范围内的第二级缓存   22.4.2 配置集群范围内的第二级缓存   22.4.3 在应用程序中管理第二级缓存   22.4.4 Session与第二级缓存的交互模式  22.5 运行本章的范例程序  22.6 小结  22.7 思考题 第23章 管理Session和实现对话  23.1 管理Session对象的生命周期   23.1.1 Session对象的生命周期与本地线程绑定   23.1.2 Session对象的生命周期与JTA事务绑定  23.2 实现对话   23.2.1 使用游离对象   23.2.2 使用手工清理缓存模式下的Session  23.3 小结  23.4 思考题 第24章 Hibernate与Struts框架  24.1 实现业务数据  24.2 实现业务逻辑  24.3 netstore应用的订单业务  24.4 小结 第25章 Hibernate与EJB组件  25.1 创建EJB组件   25.1.1 编写Remote接口   25.1.2 编写Home接口   25.1.3 编写Enterprise Java Bean  25.2 在业务代理中访问EJB组件  25.3 发布J2EE应用   25.3.1 在JBoss上部署EJB组件   25.3.2 在JBoss上部署Web应用   25.3.3 在JBoss上部署J2EE应用  25.4 小结 附录A 标准SQL语言的用法  A.1 数据完整性   A.1.1 实体完整性   A.1.2 域完整性   A.1.3 参照完整性  A.2 DDL数据定义语言  A.3 DML数据操纵语言  A.4 DQL数据查询语言   A.4.1 简单查询   A.4.2 连接查询   A.4.3 子查询   A.4.4 联合查询   A.4.5 报表查询 附录B Java语言的反射机制  B.1 Java Reflection API简介  B.2 运用反射机制来持久化Java对象 附录C 用XDoclet工具生成映射文件  C.1 创建带有@hibernate标记的Java源文件  C.2 建立项目的目录结构  C.3 运行XDoclet工具 附录D 发布和运行netstore应用  D.1 运行netstore所需的软件  D.2 netstore应用的目录结构  D.3 安装SAMPLEDB数据库  D.4 安装和配置JBoss服务器  D.5 发布netstore应用   D.5.1 在工作模式1下发布netstore应用   D.5.2 在工作模式2下发布netstore应用  D.6 运行netstore应用 附录E Hibernate 3升级指南  E.1 Hibernate API 变化   E.1.1 包名   E.1.2 org.hibernate.classic包   E.1.3 Hibernate所依赖的第三方软件包   E.1.4 异常模型   E.1.5 Session接口   E.1.6 createSQLQuery()   E.1.7 Lifecycle 和 Validatable 接口   E.1.8 Interceptor接口   E.1.9 UserType和CompositeUserType接口   E.1.10 FetchMode   E.1.11 PersistentEnum   E.1.12 对Blob 和Clob的支持   E.1.13 Hibernate中供扩展的API的变化  E.2 元数据的变化   E.2.1 检索策略   E.2.2 对象标识符的映射   E.2.3 集合映射   E.2.4 DTD  E.3 查询语句的变化  E.4 把Hibernate 2应用升级到Hibernate 3应用

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值