TaskInstance taskInstance =
jbpmContext.loadTaskInstanceForUpdate(taskInstanceId); taskInstance.end(); } finally {
jbpmContext.close(); }
正象是后台信息一样,下面部分解释了Jbpm是怎样管理持久化和使用Hibernate的。 JbpmConfiguration维护了一组服务工厂(ServiceFactory),服务工厂象前面内容所说的在jbpm.cfg.xml中配置,并且是lazy(懒)实例化的。当在第一次需要
DbpersistenceServiceFactory时它被实例化,然后服务工厂就被在JbpmConfiguration中维护,DbpersistenceServiceFactory管理一个hibernate SessionFactory,而hibernate SessionFactory也是在第一次被请求的时候才创建的。
图 7.2 持久化相关类
在调用jbpmConfiguration.createJbpmContext()方法期间,只有JbpmContext被创建,这时没有更多的持久化相关实例。JbpmContext管理一个DbPersistenceService,它在被第一次请求的时候实例化。DbPersistenceServece管理hibernate会话(session),
hibernate会话在DbPersistenceServece中也是被Lazy(懒)创建的,因此结果是:一个hibernate会话只有在第一个请求持久化的操作被调用时才被打开,而在这之前不会。
7.1.3 高级API用法
DbPersistenceService维护了一个懒实例化的hibernate 会话,所有的数据库存取都是通过这个hibernate会话完成的。所有的查询和更新都是通过Jbpm暴露出的xxxSession类完成的,例如GraphSession、SchedulerSession、LoggingSession等等,这些session类都是提交hibernate查询并且都使用下层相同的hebernate会话。
XxxxSession类也是通过JbpmContext来获取的。
7.2 配置持久化服务
7.2.1 The hibernate session factory
默认情况下,DbpersistenceServiceFactory将使用classpath根下的资源文件hibernate.cfg.xml来创建hibernate会话工厂。注意:hibernate资源配置文件由“jbpm.hibernate.cfg.xml”属性指定,可以在jbpm.cfg.xml中定制。下面是默认配置:
...
当属性“resource.hibernate.properties”被指定,则指定的资源文件中的属性将会覆盖hibernate.cfg.xml中的配置,因此我们可以使用hibernate.properties文件、而不是更新hibernate.cfg.xml文件来方便的更新Jbpm指向自己的数据库:hibernate.cfg.xml仅仅需要拷贝过去不需要做任何更改。
7.2.2 The DbPersistenceServiceFactory
DbpersistencrServiceFactory拥有三个配置属性:isTransactionEnabled、sessionFactoryJndiName和dataSourceJndiName。要指定这几个属性中的任何一个,你需要象bean一样在factory元素中定义服务工厂,如下:
factory=\
l isTransactionEnabled:默认情况下,Jbpm将会开始并且结束hibernate事
务。如果禁用事务并且禁止使用hibernate管理事务,可以象上例一样配置isTransactionEnabled属性为false。更多信息,请参考“第7.3节 hibernate事务” 。
l sessionFactoryJndiName:默认情况下这个属性为空,意味着会话工厂不从
JNDI获取。如果设置了,当需要使用会话工厂创建一个hibernate会话时,则将会从所提供的JNDI名称中通过jndi获取会话工厂。 l dataSourceJndiName:默认情况下这个属性为空,JDBC连接的创建会委托给
hibernate。通过指定一个datasource,当打开一个新的会话时,Jbpm将会从datasource中获取一个JDBC连接并且把这个连接提供给hibernate。有关用户提供JDBC连接,请参考“第7.5节用户提供的素材”。
7.3 Hibernate事务
默认情况下,Jbpm将委托事务到hibernate,并且使用每个事务一个会话的模式。Jbpm在hibernate会话被打开时启动一个hibernate事务,这在第一次调用jbpmContext的一个持久化操作时发生,事务会在hiberante会话关闭之前被提交,这在jbpmContext.close()中发生。
使用jbpmContext.setRollbackOnly()回滚一个事务,在这种情况下,事务会在jbpmContext.close()方法中关闭会话之前回滚。
要禁止Jbpm在hibernate API之上调用任何事务,可以设置isTransactionEnabled属性为false,就象上面的“第7.2.2 节 The DbPersistenceServiceFactory”说明的一样。
7.4 管理事务
当在J2EE应用服务器中(如JBOSS)使用Jbpm时,大多情况下需要管理事务。如:
l 在应用服务器中配置一个DataSource
l 配置hibernate使它的连接使用DataSource l 使用容器管理事务 l 在Jbpm中禁用事务
7.5 用户提供的素材
你也可以编程向Jbpm提供JDBC连接、hibernate会话和hibernate会话工厂。 当这些资源提供给Jbpm时,Jbpm会使用所提供的资源,而不是配置中指定的。 JbpmContext类包含了一些方便的方法来实现通过编程注入资源。例如,向Jbpm提供一个JDBC连接,可以使用如下代码:
JbpmContext jbpmContext =
jbpmConfiguration.createJbpmContext();
try {
Connection connection = ...;
jbpmContext.setConnection(connection);
// invoke one or more persistence operations
} finally {
jbpmContext.close(); }
JbpmContext类有如下方便的方法用来支持编程提供资源:
JbpmContext.setConnection(Connection); JbpmContext.setSession(Session);
JbpmContext.setSessionFactory(SessionFactory);
7.6 自定义查询
Jbpm使用的所有HQL查询都集中在一个配置文件中,这个资源文件在hibernate.cfg.xml配置文件中被引用,如下:
要自定义一个或更多查询,先拷贝一个原始文件并且把你自定义的版本放在classpath下的某个位置,然后在hibernate.cfg.xml中更新
“org/jbpm/db/hibernate.queries.hbm.xml”,使之指向你自己定义的版本。
7.7 数据库兼容
Jbpm可以在hibernate所支持的任何数据库上运行。
Jbpm中的示例配置文件(src/config.files)指定了使用的是hypersonicin in-memory数据库,这个数据库在开发和测试阶段是一个理想的选择,hypersonicin in-memory数据库把数据放在内存中而不是存储到磁盘。
7.7.1 改变Jbpm数据库
下面列出了当改变Jbpm使用一个不同数据库时的作法:
l 把jdbc驱动库放在classpath
l 改变Jbpm所使用的hibernate配置 l 在新的数据库中创建表
7.7.2 Jbpm数据库表
Jbpm.db子项目包含了一系列的驱动库、说明和脚本来帮助你在你所选择的数据库上开始工作。请查阅jbpm.dm工程根目录下的readme.html文件来获取更多信息。
尽管Jbpm有能力生成所有数据库的DDL脚本,但是这些不总是最优的,因此你可能想要你的DBA查阅这些生成的DDL,来进行列类型的优化和使用索引。
在开发中,你可能对下面的hibernate配置感兴趣:如果你设置hibernate配置属性“hibernate.hbm2ddl.auto”为“create-drop”(例如在hibernate.cfg.xml中),则在应用中第一次使用时,便会在数据库中自动创建,当应用关闭时,则表会被删除。
也可以编程调用jbpmConfiguration.createSchema()和jbpmConfiguration.dropSchema()方法来实现表的创建和删除。
7.8 结合你自己的hibernate类
在你自己的项目中,你可能使用hibernate来持久化,可以使你自己的持久化类与Jbpm持久化类结合,这是可选的。这样做有两个主要的好处:
首先,会话、连接和事务管理变得容易。通过结合Jbpm和你自己的持久化在一个hibernate会话工厂,则只有一个hibernate会话、一个jdbc连接来处理你自己的和Jbpm的持久化,因此,Jbpm的更新与你自己的域模型更新在相同的事务中,这可以排除再使用一个事务管理器。
其次,这毫无争议的就把你自己的hibernate持久化对象融入了流程变量中。
可以通过创建一个核心的hibernate.cfg.xml来很容易的与Jbpm持久化类集成你自己的持久化类,你可以使用Jbpm中的src/config.files/hibernate.cfg.xml作为开始,然后在其中添加你自己的hibernate映射文件的引用。
7.9 定制Jbpm的hibernate映射文件
你可以按照如下方式来定制任何Jbpm的hibernate映射文件:
l 从源码(src/java.jbpm/…)或Jbpm库中拷贝Jbpm的hibernate映射文件 l 把拷贝的文件放置到classpath下任何你想要放置的地方
l 在hibernate.cfg.xml配置文件中更新指向自定义的映射文件
7.10 二级缓存
流程定义加载一次之后,Jbpm使用hibernate的二级缓存来把它放在内存中,流程定义类和集合在Jbpm的hibernate映射文件中使用cache元素来配置,如下:
因为流程定义不会改变(应该不会改变),所以把它们放在二级缓存中是很好的做法,请参考“第16.1.3节 改变已发布的流程定义”。 二级缓存是JBOSS Jbpm实现的一个很重要的方面,如果没有这个缓存,JBOSS Jbpm在与其它实现BPM引擎的技术进行比较时,会是一个严重的缺陷。 缓存策略设置为nonstrict-read-write。在运行时,缓存策略可以设置为read-only,在这种情况下,你需要一个单独的hibernate映射文件来部署流程,这也是为什么我们选择nonstrict-read-write的原因。