其实,在EJB中比较复杂的Bean不是会话Bean,不是消息驱动Bean,而是实体Bean。因为它涉及到对象序列化,O/R Mapping等一些持久化技术(还有一直没有流行起来的对象数据库)。
什么是实体Bean?
实体Bean是有着一组属性并且每个属性与数据库表中的每个字段一一对应并且公开get和set方法供外界访问。如图:
每一个属性对应数据库表中的一个字段,这样一个Bean实例就对应了表中的一条记录。这里要注意的是,并不是固定的一个实例对应一条记录。如表中有5条记录,不一定有5个实体Bean实例来对应,有可能3个,有可能8个。为什么呢?
1.因为对象的创建和删除是十分消耗系统资源的,所以不可以为成千上万条记录创建成千上万个对象。解决的办法是将被实例化的Bean重复再利用,这样就可以节省这样的开销。容器可以动态的分配实体Bean实例给不同的客户端的EJB对象使用,这样不但节省了容器不必要地实例化Bean的开销,而且节省了系统资源。
2.因为EJB规定只能有一个线程可以运行在一个Bean实例中,也就是说会话Bean,消息驱动Bean,实体Bean都是单线程的。所以为了使不同的客户端方便的访问相同的数据,容器将实例化相同实体Bean的多个实例供不同的客户端使用。问题来了,当使用这种方法时,怎样保证数据的同步呢?假如其中一个客户更改这条数据,如何反映到持有相同实体Bean的不同实例的客户端呢?为了达到实体Bean实例缓存的一致性,每个实体Bean都必须公开两个方法,ejbLoad()和ejbStore()。容器通过调用ejbLoad()方法从存储空间(数据库,文件等)中读取数据。调用ejbStore()方法保存数据到存储空间中。
实体Bean实例是跟数据库表中的数据一一对应的。可以把它就看成关系型数据在JAVA中的表现,因为那没有什么区别,改变它就改变了数据库中的数据。实体Bean是对应到数据库中的视图,对象和数据的同步是由EJB容器完成的(调用ejbLoad和ejbStore方法)。容器在实体Bean挂起前调用ejbStore方法,在实体Bean激活之后调用ejbLoad方法。
如何保持实体Bean?
1.可以手工完成持久化操作,也就是说,可以编写代码将内存中的字段转换成数据库中的字段。这样,就不得不处理一些如存储,导入和在实体Bean中查找数据的持久化方法。而且必须编写持久化API,如JDBC或SQL/J等。实体Bean可以通过JDBC执行SQL INSERT语句在数据库中插入数据。也可以通过SQL DELETE语句从数据库中删除数据。
2.可以让EJB容器完成持久化操作。容器会自动生成数据库访问代码,执行SQL语句。利用容器管理的持久,我们就可以不用编写JDBC代码了。解放了:)
实体Bean是怎样创建和删除的?
在EJB中,客户端是不能直接调用Bean的。它们通过调用EJB对象来执行操作。如下:
客户端在EJB对象或Home对象上调用remove方法删除实体Bean数据。ejbRemove()方法并不是将实体Bean从内存中删除,只是删除数据库中的数据。因为可以重复利用这个Bean实例操作不同的数据库数据。ejbRemove方法是每个实体Bean必须的方法,它不接收参数。如果客户端终止连接,并不会调用ejbRemove方法,因为一个实体Bean的存在时间比客户端会话时间要长。实体Bean是可以被查找的。利用定位器方法。
实体Bean的上下文
所有的企业Bean都有一个上下文(Context)对象来识别Bean所处的环境。这些上下文中包括EJB容器设置的环境信息。Bean可以访问这个上下文得到信息。实体Bean上下文中的getEJBLocalObject()方法可以得到当前客户端与实体Bean关联的EJB对象(容器产生的EJB对象)。getEJBObject()方法可以得到EJB本地对象。最重要的方法是getPrimaryKey(),它可以得到实体Bean实例的主键。主键可以唯一标识实体Bean实例,因为两个实体Bean数据库中的数据是不可能有相同的主键的。确定实例关联哪个数据库数据时就可以调用该方法。当实体Bean挂起一段时间被激活后,实体Bean实例必须对上下文对象执行一个getPrimaryKey()方法调用来确定它将要映射处理的数据库中的哪些数据。