1.缘由

有一庞大Python django webproject,要引入工作流引擎,像OA一样。方便的流程控制与管理。Python或django关于工作流的开源插件,稀少,并且弱爆了,终于选用java的开源框架JBPM。为了使Pythonproject整合Java框架,尝试过jbpm-console提供的 REST, 以失败告终,终于选用Python 直接调用Java代码。操作JBPM.

1.1pythonprojectREST与JBPM Workbench交互 失败

REST URL格式

http://host:8080/jbpm-console/rest/runtime/{deploymentId}/process/{processDefId}/start

比如

http://localhost:8080/jbpm-console/rest/runtime/com.fun:test:1.0.0/process/test.TaskData/start

并不是如文档所言以 /runtime 打头。REST接口是jbpm-console.war子项目提供的,必须部署此项目。

失败原因:流程复杂而且须要处理复杂数据,JBPM内部能够建立数据模型,使用POJO传递或存储复杂数据,可是REST接口无法POST某种格式的数据。使jbpm-console自己主动映射成POJO,假设自己写java代码实现REST 数据转POJO,可行度再说,既然都自己写Java代码了。就不是必需走REST,网络慢。不如直接Java代码完毕JBPM逻辑。

比如:JBPM创建了一个数据模型Person,形如

public class Person{
String name = null;
int age = 0;
boolean isMan= false;
}

REST接口没法POST数据,使JBPM能映射成一个Person类的实例,默认jbpm-console会当作三个零散变量(參数)处理,不会当一个对象的三个属性。所以默认会在数据库里variableinstancelog表存入三行记录,每行代表一个变量。达不到期望结果。仅仅有一条记录,仅仅存一个对象的序列化信息。

假设某大神知道怎样POST数据(或Python简单处理POST数据后再发请求),jbpm-console能自己主动映射成POJO。请留言。铭谢!

2.Python 直接调用 Java

2.1 jPype 不採用

  • 坑(能够人为避免)

    • 当Java方法或属性是Pythonkeyword时,无法訪问
    • 无法訪问不在包以下(default包)的Java代码
  • 缺点

    • 适合Python26。实測27版本号启动jvm失败

    • Java特殊方法不支持,如java.lang.Class.forName(String classname),硬伤(由于此方法在框架/容器代码中十分频繁)

    • 无法实例化匿名内部类。仅能实例化静态内部类

    • 线程同步:明白支持synchronized类同步,notify, notifyAll, wait等未提及怎样处理

    • 明白表示有较大性能影响:can impose rather large performance bottlenecks

2.2 pyjnius  採用

  • 长处

    • 封装出色。文档未提及也未发现有特殊Java语法或高级机制限制(明白支持重载)

    • 大量运用于Python自己主动化測试安卓应用,与Java语言亲和度更高

    • 支持Python继承/实现java的抽象类/接口(接口中静态成员不被支持)

    • 提供一个近乎万能的autoclass。反射出一个Java类的代理(能在Python层面实现Java的代理类)

  • 缺点

    • windows安装十分困难,无预编译版本号,预计必须GCC(MinGW, vs08, vs10均失败)

3.python<-->java se <-->jbpm <-->hibernate<-->mysql原理简述

  • use pyjnius connect python and java

  • java se code uses CLASSPATH include jbpm jars and hibernate jars(must do before import pyjnius)

  • jbpm uses hibernate in two step

    • jbpm uses jBPM.proporties to setup and register DataSource in specified db(we specify mysql)

    • jbpm calls hibernate to operate db, but hard code to act like use H2 db, we force hibernate act like use mysql by passing custom params when create it's entiy manager(more details in code)

  • hibernate operates db, I have find a way to directly control it's behavior completely, transparently

  • Codes speak louder. More details in example.



4. 附Python和Java代码

完整project代码地址http://download.csdn.net/detail/secretx/7814991



3.1Python代码

#coding=utf8

import os
javapath = "/home/luodh/workspace/jbpm-6.0.0.Final-bin/*:/home/luodh/workspace/jbpm-6.0.0.Final-bin/lib/*"
os.environ['CLASSPATH'] = javapath """call java code directly"""
from jnius import autoclass ProcessMain = autoclass('com.sample.ProcessMain') #.号分隔
ProcessMain.pythonCallMeStatic() #ok, before ProcessMain(), can call java static method
# ProcessMain.pythonCallMeInstance() #error, before ProcessMain(), can not call java instance method process_main = ProcessMain()
process_main.pythonCallMeInstance()
process_main.pythonCallMeStatic() ProcessMain.pythonCallMeStatic()
ProcessMain.pythonCallMeInstance() # after ProcessMain(), python does not distinguish static/instance method
print "-----------------------------------------" """use a python proxy class"""
from jnius import JavaClass, MetaJavaClass, JavaMethod, JavaField class PythonProxyJavaClass(JavaClass):
__javaclass__ = "com/sample/ProcessMain" # /分隔.
__metaclass__ = MetaJavaClass # must be this
# __javaconstructor__ = ( # 构造函数,源于反编译
# "()V",
# )
# 构造函数规则尚未确定,文档有误
pythonCallMeStatic = JavaMethod("()V", static=True)
pythonCallMeInstance = JavaMethod("()V")
s1 = JavaField("Ljava/lang/String;", static=True)
s2 = JavaField("Ljava/lang/String;") PythonProxyJavaClass.pythonCallMeStatic()
# PythonProxyJavaClass.pythonCallMeInstance() # 实例化对象之前,不可调用实例方法 PythonProxyJavaClass().pythonCallMeStatic() # 实例化对象之后,python不区分静态(类)方法和非静态(实例)方法
PythonProxyJavaClass().pythonCallMeInstance() PythonProxyJavaClass.pythonCallMeStatic()
PythonProxyJavaClass.pythonCallMeInstance() """python严格区分静态/非静态的Field"""
print PythonProxyJavaClass.s1
# print PythonProxyJavaClass.s2 # error, 类不与成员变量绑定
print PythonProxyJavaClass().s1 # none,实例不与静态变量绑定
print PythonProxyJavaClass().s2



3.2 Java代码(jbpm相关)

package com.sample;

import java.util.HashMap;

import java.util.List;

import java.util.Map;

import java.util.Properties;

import javax.persistence.EntityManagerFactory;

import javax.persistence.Persistence;

import org.jbpm.test.JBPMHelper;

import org.kie.api.KieBase;

import org.kie.api.KieServices;

import org.kie.api.runtime.KieContainer;

import org.kie.api.runtime.KieSession;

import org.kie.api.runtime.manager.RuntimeEngine;

import org.kie.api.runtime.manager.RuntimeEnvironmentBuilder;

import org.kie.api.runtime.manager.RuntimeManager;

import org.kie.api.runtime.manager.RuntimeManagerFactory;

import org.kie.api.task.TaskService;

import org.kie.api.task.model.TaskSummary;

public class ProcessMain {

	public static String s1 = "I'm static field";

	public String s2 = "I'm instance field";

	public static RuntimeManager manager = null;

	public static void main(String[] args) {

		callJBPM(false);

		System.exit(0);

	}

	public void pythonCallMeInstance(){

		callJBPM(true);

	}

	public static void pythonCallMeStatic(){

		callJBPM(true);

	}

	public static void callJBPM(boolean fake){

		if (fake){

			System.out.println("I'm java, python call me to play jbpm. I don't know how to include jbpm jars.");

			return;

		}

		KieServices ks = KieServices.Factory.get();

		KieContainer kContainer = ks.getKieClasspathContainer();//载入kmodule.xml

		KieBase kbase = kContainer.getKieBase("kbase");// 依据暴露的名字(kbase标签的name属性)获取知识库对象

		if(manager == null) {

			manager = createRuntimeManager(kbase);// use synchronized static block

		}

		RuntimeEngine engine = manager.getRuntimeEngine(null);

		KieSession ksession = engine.getKieSession();

		TaskService taskService = engine.getTaskService();// engine一共3接口 getAuditLogService

		ksession.startProcess("com.sample.bpmn.hello");

		// let john execute Task 1

		List<TaskSummary> list = taskService.getTasksAssignedAsPotentialOwner("john", "en-UK");

		TaskSummary task = list.get(0);

		System.out.println("John is executing task " + task.getName());

		taskService.start(task.getId(), "john");

		taskService.complete(task.getId(), "john", null);

		// let mary execute Task 2

		list = taskService.getTasksAssignedAsPotentialOwner("mary", "en-UK");

		task = list.get(0);

		System.out.println("Mary is executing task " + task.getName());

		taskService.start(task.getId(), "mary");

		taskService.complete(task.getId(), "mary", null);

		manager.disposeRuntimeEngine(engine);

//		System.exit(0);

	}

	private static RuntimeManager createRuntimeManager(KieBase kbase) {

		//因为要获取task,须要TaskService进行入库操作,需提前启动数据库,准备数据源

//		JBPMHelper.startH2Server();//不用h2,则不启动

		JBPMHelper.setupDataSource();

		Map<String, String> map = configHibernate();

		EntityManagerFactory emf = Persistence.createEntityManagerFactory("org.jbpm.persistence.jpa", map);

		RuntimeEnvironmentBuilder builder = RuntimeEnvironmentBuilder.Factory.get()

			.newDefaultBuilder().entityManagerFactory(emf)

			.knowledgeBase(kbase);

		return RuntimeManagerFactory.Factory.get()

			.newSingletonRuntimeManager(builder.get(), "com.sample:example:1.0");//any_non_blank_string

	}

	private static Map<String, String> configHibernate(){

		Map<String, String> map = new HashMap<String, String>();

		Properties properties = new Properties();

        try {

            properties.load(JBPMHelper.class.getResourceAsStream("/jBPM.properties"));

        } catch (Throwable t) {

            // do nothing, use defaults

        }

		map.put("hibernate.dialect", properties.getProperty("hibernate.dialect", "org.hibernate.dialect.MySQL5Dialect"));

		map.put("hibernate.hbm2ddl.auto", properties.getProperty("hibernate.hbm2ddl.auto", "update"));

		map.put("hibernate.show_sql", properties.getProperty("hibernate.show_sql", "false"));

		map.put("hibernate.format_sql", properties.getProperty("hibernate.format_sql", "true"));

		return map;

	}

}

python调用Java代码,完毕JBPM工作流application的更多相关文章

  1. Python调用Java代码部署及初步使用

    Python调用Java代码部署: jpype下载地址:https://www.lfd.uci.edu/~gohlke/pythonlibs/#jpype 下载的时候需要使用Chrome浏览器进行下载 ...

  2. python 调用 java代码

    一.JPype简述 1.JPype是什么? JPype是一个能够让 python 代码方便地调用 Java 代码的工具,从而克服了 python 在某些领域(如服务器端编程)中的不足. 2.JPype ...

  3. python调用java代码 java虚拟机(jvm)

    1.新建com文件夹,在里面新建 fibnq.java package com; public class fibnq { public fibnq(){} public int fb(int n){ ...

  4. python调用Java代码

    #coding:utf-8 #!/usr/bin/python from jpype import * import os.path,json from ethereum.utils import e ...

  5. python调用java代码,jpype简单使用

  6. python2.7 (x64) 调用 java 代码实践

    背景:公司实施接口自动化测试,使用的框架python+unittest:因为开发使用的架构师SpringBoot,在测试过程中难免需要使用到python调用JAVA的一些接口,所以需要用到python ...

  7. C#调用java代码(IKVMC)

    参考资料:https://blog.csdn.net/threadroc/article/details/51406587 参考1:http://www.cnblogs.com/Jack-Blog/p ...

  8. 使用 jpype 库实现 Python 调用 java 的 jar 包中的功能

    一.what's the JPype JPype 是一个能够让 python 代码方便地调用 Java 代码的工具.在某些时候 java 的能力更强,我们可以用 java 写一个模块的功能然后用 Py ...

  9. python - jpype模块,python调用java的接口

    转载自: http://www.cnblogs.com/junrong624/p/5278457.html https://www.cnblogs.com/fanghao/p/7745356.html ...

随机推荐

  1. Unity Image Effect

    1.工作原理 Image Effect是运用于一个Camera,将Camera视见体最终看到的二维图像内容作为一个2DTexture传递给一个Shader, 然后在Shader的Fragment渲染阶 ...

  2. Python开源爬虫项目代码:抓取淘宝、京东、QQ、知网数据--转

    数据来源:数据挖掘入门与实战  公众号: datadw scrapy_jingdong[9]- 京东爬虫.基于scrapy的京东网站爬虫,保存格式为csv.[9]: https://github.co ...

  3. <Sicily>Brackets Matching

    一.题目描述 Let us define a regular brackets sequence in the following way: Empty sequence is a regular s ...

  4. 使用sysbench 对mysql进行性能测试

    使用sysbench 对mysql进行性能测试 sysbench是一个开源的.模块化的.跨平台的多线程性能测试工具,可以用来进行CPU.内存.磁盘I/O.线程.数据库的性能测试.目前支持的数据库有My ...

  5. bzoj 2456: mode 思维题 好题

    题目描述: 给你一个 $n$ 个数的数列,其中某个数出现了超过 n div 2 次即众数,请你找出那个数.空间大小:1mb 题解:显然,我们是不能开任何数组的,此题专卡空间.然而我们要求的东西也十分简 ...

  6. ELK安装笔记

    1.jdk安装 2.logstash安装使用 #命令方式[root@ELK ELK]# tar xf logstash-5.3.2.tar.gz [root@ELK logstash-5.3.2]# ...

  7. 洛谷1726 上白泽慧音 tarjan模板

    题目描述 在幻想乡,上白泽慧音是以知识渊博闻名的老师.春雪异变导致人间之里的很多道路都被大雪堵塞,使有的学生不能顺利地到达慧音所在的村庄.因此慧音决定换一个能够聚集最多人数的村庄作为新的教学地点.人间 ...

  8. uname---用于打印当前系统相关信息

    uname命令用于打印当前系统相关信息(内核版本号.硬件架构.主机名称和操作系统类型等). 语法 uname(选项) 选项 -a或--all:显示全部的信息: -m或--machine:显示电脑类型: ...

  9. 使用maven安装jar到本地仓库

    mvn install:install-file "-DgroupId={安装的jar包的groupid,可以随意起名}" "-DartifactId={安装jar包的I ...

  10. 【习题 8-16 UVA - 1618】Weak Key

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 枚举N[q]和N[r]的位置 因为N[q]是最大值,且N[r]是最小值. 且它们是中间的两个. 枚举这两个可以做到不重复枚举. 然后 ...