EJB (Entity Enterprise JavaBeans) 是一种可以把持久性数据映射到Java组件上的简便方法。CMP (Container-Managed persistence)提供了快速开发功能,这是因为EJB 容器可自动处理持久性数据的加载和存储。然而,在具有许多优点的同时,假如Entity EJB没有正确使用,也会导致性能的大幅下降。本文具体介绍了几个常见的编程缺陷,它们经常使EJB的程序员犯错,并妨碍其实体(Entity)beans的性能。
Primary Key类
类似于数据库中的行,实体beans有一个主键(primary key)与它关联。这个主键可以是实体bean的一个单一字段。在这种情况下,实体bean可以用字段的类作为主键。
还可能提供一种自定义的主键类。对于复合主键来说,必须定制一个主键类,来映射多个实体bean的字段。
使用定制的主键类,开发人员必须实现hashCode和equals方法。因为EJB容器常在其内部数据结构中使用主键类,所以这个类必须正确和有效的实现hashCode和equals方法 (参见清单1)。
清单 1:
一个低效但正确的主键类
public class MyPk
implements java.io.Serializable
{
public String str;
public int i;
public byte b;
public MyPk() {}
public int hashCode() { return -1; }
public boolean equals(Object o) {
if ((o != null) && (MyPk.class.equals(o.getClass()))) {
MyPk other = (MyPk) o;
return other.str.equals(str) && other.i == i && other.b == b;
} else {
return false;
}}
}
实现hashCode方法
hashCode方法对于两个equal的对象,必须返回相同的值,而且应该相对均匀地分配哈希值。下面显示的第一种实现方法正确而有效,但是根本没有分配哈希值。这个hashCode实现把全部哈希表变换到一个列表中,而且必须线性检索。显然,这样违反了可检索性数据结构的设计初衷。
private int hash = -1;
public int hashCode() {
if (hash == -1) {
hash = str.hashCode() ^ i ^ b;
}
return hash;
}
上面的hashCode实现计算了字符串的哈希值和原字段的异或(XOR)值。 与其它的逻辑运算符相比,诸如AND和OR,XOR应该是更可取的,因为它可以更好地分配哈希值。这种实现还可以把哈希值缓存在一个成员变量中,以避免重复计算这个值。
实现Equals 方法
equals方法的功能是使用传入的参数比较当前对象,假如对象有相同的值,就返回true。默认的java.lang.Object.equals用于比较引用(指针)值,假如它们相等就返回true。对于大多数的主键类,需要重写这个方法,以便在主键类中比较这些值(参见清单 2)。
清单:2
一个有效的equals实现
public final class MyPk ...
public boolean equals(Object o) {
if (o == this) return true;
if (o instanceof MyPk) {
MyPk other = (MyPk) o;
return other.hashCode() == hashCode() &&
other.i == i && other.b == b &&
other.str.equals(str);
} else {
return false;
}}
这是一种优化的equals实现,它的第一行用与此相反的方式比较传入的引用。第一,虽然这看起来有点生疏,但这是EJB容器检查一个主键是否已经在它的数据结构中存在的常用方法。
第二,我们已经用一个更有效的检查实例替代了getClass().equals。假如传入参数的类是MyPk类或它的一个子类,操作符的实例将返回true。 用final修饰MyPk类,这样创建的方法可以安全地使用操作符的实例,因为这样就不存在子类了。
最后,比较哈希表和成员变量。Java中的表达式具有短路功能,这意味着假如第一个表达式是false,第二个表达式将不再计算。这个equals方法很好的利用了这一点,先用最简易的比较调整了and语句的顺序。在这个例子里,首先比较的是哈希值,这是因为我们的实现缓存了这个值,而且发生两个对象具有相同的hashCode但却不相等的情况很少。接下来比较的是原始字段;最后是调用花费资源最多的java.lang.String.equals。
加载和存储实体beans