首页 > 学院 > 开发设计 > 正文

利用实体EJB来避免性能缺陷

2019-11-18 13:49:00
字体:
来源:转载
供稿:网友

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



发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表