JBPM4常用错误汇总 收藏

简介:
JBPM4常用错误汇总 收藏 http://blog.csdn.net/cnham/archive/2009/12/16/5013068.aspx 
1.在tomcat6.0下布署错误 
   基于JBPM4的web项目jsp页面发布出错 
现象: 
   javax.servlet.ServletException: java.lang.LinkageError: loader constraint violation: when resolving interface method "javax.servlet.jsp.JspApplicationContext.getExpressionFactory()Ljavax/el/ExpressionFactory;" the class loader (instance of org/apache/jasper/servlet/JasperLoader) of the current class, org/apache/jsp/OnDuty/wfmanage_jsp, and the class loader (instance of org/apache/catalina/loader/StandardClassLoader) for resolved class, javax/servlet/jsp/JspApplicationContext, have different Class objects for the type javax/el/ExpressionFactory used in the signature 
       org.apache.jasper.servlet.JspServlet.service(JspServlet.java:275) 
       javax.servlet.http.HttpServlet.service(HttpServlet.java:717) 
。。。 

原因: 
   项目中WEB-INF\lib中的三个jar包(juel.jar, juel-engine.jar, juel-impl.jar)和tomcat6下lib中jar包(el-api.jar)冲突 

解决方法: 
   方法一:换成tomcat5.5 一点问题也没有了 
   方法二:将juel.jar, juel-engine.jar, juel-impl.jar这三个包复制到tomcat6下lib中,并删除原来的el-api.jar,切记要把WEB-INF\lib中的juel.jar, juel-engine.jar, juel-impl.jar删除。不然还是要冲突。 

2.无法保存(布署)含有中文的流程定义文件 
现象: 
保存流程定义文件 
   <?xml version="1.0" encoding="GBK"?> 
   <process name="leave" xmlns="http://jbpm.org/4.0/jpdl"> 
      <start g="201,14,48,48" name="开始"> 
         <transition g="-42,-10" name="请假" to="填写请假单"/> 
      </start> 
      ... 
提示错误: 
   MalformedByteSequenceException:Invalid byte 1 of 1-byte UTF-8 sequence 

原因: 
   XML字符串大概经过了下面一些方法才被解析为DOM: 
org.jbpm.pvm.internal.repository.DeploymentImpl: 
  public NewDeployment addResourceFromString(String resourceName, String text) {  
      addResourceFromStreamInput(resourceName, new StringStreamInput(text));  
      return this;  
  }  
public NewDeployment addResourceFromString(String resourceName, String text) { 
  addResourceFromStreamInput(resourceName, new StringStreamInput(text)); 
  return this; 


org.jbpm.pvm.internal.stream.StringStreamInput: 
   public class StringStreamInput extends StreamInput {  
      String string;  
      public StringStreamInput(String string) {  
        this.name = "string";  
        this.string = string;  
      }  
     public InputStream openStream() {  
       byte[] bytes = string.getBytes();  
       return new ByteArrayInputStream(bytes);  
     }  
   }  
public class StringStreamInput extends StreamInput { 
  String string; 
  public StringStreamInput(String string) { 
    this.name = "string"; 
    this.string = string; 
  } 
  public InputStream openStream() { 
    byte[] bytes = string.getBytes(); 
    return new ByteArrayInputStream(bytes); 
  } 


org.jbpm.pvm.internal.xml.Parse: 
    protected InputSource getInputSource() {  
      if (inputSource!=null) {  
        return inputSource;  
      }  
      if (streamInput!=null) {  
        inputStream = streamInput.openStream();  
        return new InputSource(inputStream);  
      }  
     addProblem("no source specified to parse");  
     return null;  
   }  
protected InputSource getInputSource() { 
  if (inputSource!=null) { 
    return inputSource; 
  } 
  if (streamInput!=null) { 
    inputStream = streamInput.openStream(); 
    return new InputSource(inputStream); 
  } 
  addProblem("no source specified to parse"); 
  return null; 


org.jbpm.pvm.internal.xml.Parser: 
    protected Document buildDom(Parse parse) {  
      Document document = null;  
      try {  
        SAXParser saxParser = saxParserFactory.newSAXParser();  
        XMLReader xmlReader = saxParser.getXMLReader();  
        // ...  
          
       InputSource inputSource = parse.getInputSource();   
       xmlReader.parse(inputSource);  
     
     } catch (Exception e) {  
       parse.addProblem("couldn't parse xml document", e);  
     }  
     
     return document;  
   }  
   经过层层包装、拆包、再包装再拆包,可怜的字符串终于来到SAX解析器的手上。问题是jBPM在中间调用了String.getBytes():这个方法会把Java字符串(Unicode)转换为系统默认编码并返回对应的byte[],但当InputSource中没有设置编码信息时,SAXParser默认是以UTF-8编码来读取输入流的。我的开发机的系统默认编码是GBK,于是就出问题了。 

解决方法: 
   String xmlStr = "<?xml version=\"1.0\" encoding=\"" + System.getProperty("file.encoding") + "\"?><test name=\"名称\"></test>"; 
   这里就用System.getProperty("file.encoding")去获取系统默认编码,以便于String.getBytes()匹配。 
   如果你能确保你的WEB服务器上运行的字符集是GBK,也可以写成 
<?xml version="1.0" encoding="GBK"?> 
<process name="leave" xmlns="http://jbpm.org/4.0/jpdl"> 
   <start g="201,14,48,48" name="开始"> 
      <transition g="-42,-10" name="请假" to="填写请假单"/> 
   </start> 
   ... 

3.无法保存(布署)含有中文的流程定义文件 
现象: 
   流程定义已经成功保存到数据库中,但无法执行, 
   数据库表jbpm4_execution中的任务名称为乱码 

原因: 
   数据库中任务名称为乱码的根本原因不是hiberate保存到jbpm4_lob中的字段BLOB_VALUE造成的。而是JSP页面传递给servlet流程定义文本时,中文转码错误造成的。即servlet接收的即乱码。 

解决方法 
  1)JSP页面中显示中文乱码 
     在JSP文件中使用page命令指定响应结果的MIME类型,如<%@ page language="java" contentType="text/html;charset=gbk" %> 
  2)表单提交乱码   
    表单提交时(post和Get方法),使用request.getParameter方法得到乱码,这是因为tomcat处理提交的参数时默认的是iso-8859-1,表单提交get和post处理乱码问题不同,下面分别说明。 
    (1)POST处理 
    对post提交的表单通过编写一个过滤器的方法来解决,过滤器在用户提交的数据被处理之前被调用,可以在这里改变参数的编码方式,过滤器的代码如下: 
Java代码 
package example.util; 
    
    import java.io.IOException; 
    import javax.servlet.Filter; 
    import javax.servlet.FilterChain; 
    import javax.servlet.FilterConfig; 
    import javax.servlet.ServletException; 
    import javax.servlet.ServletRequest; 
    import javax.servlet.ServletResponse; 
    
    public class SetCharacterEncodingFilter implements Filter { 
       protected String encoding = null; 
       protected FilterConfig filterConfig = null; 
       protected boolean ignore = true; 
    
     public void destroy() { 
      this.encoding = null; 
      this.filterConfig = null; 
     } 
    
     public void doFilter(ServletRequest request, ServletResponse response, 
       FilterChain chain) throws IOException, ServletException { 
          if (ignore || (request.getCharacterEncoding() == null)) { 
       String encoding = selectEncoding(request); 
       if (encoding != null) { 
        request.setCharacterEncoding(encoding); 
       } 
      } 
    
      // Pass control on to the next filter 
      chain.doFilter(request, response); 
    
     } 
    public void init(FilterConfig filterConfig) throws ServletException { 
    
      this.filterConfig = filterConfig; 
      this.encoding = filterConfig.getInitParameter("encoding"); 
      String value = filterConfig.getInitParameter("ignore"); 
      if (value == null) { 
       this.ignore = true; 
      } else if (value.equalsIgnoreCase("true")) { 
       this.ignore = true; 
      } else if (value.equalsIgnoreCase("yes")) { 
       this.ignore = true; 
      } else { 
       this.ignore = false; 
      } 
    
     } 
    
     protected String selectEncoding(ServletRequest request) { 
    
      return (this.encoding); 
    
     } 
    
    } 

web.xml文件加入过滤器 
<filter> 
    <filter-name>Encoding</filter-name> 
    <filter-class> 
            example.util.SetCharacterEncodingFilter 
     </filter-class> 
    <init-param> 
   <param-name>encoding</param-name> 
   <param-value>gbk</param-value> 
   <!--gbk或者gb2312或者utf-8--> 
  </init-param> 
  <init-param> 
   <param-name>ignore</param-name> 
   <param-value>true</param-value> 
  </init-param> 
</filter> 
<filter-mapping> 
  <filter-name>Encoding</filter-name> 
  <servlet-name>/*</servlet-name> 
</filter-mapping> 

(2) Get方法的处理 
tomcat对post和get的处理方法不一样,所以过滤器不能解决get的乱码问题,它需要在其他地方设置。 
打开<tomcat_home>\conf目录下server.xml文件,找到对8080端口进行服务的Connector组件的设置部分,给这个组件添加一个属性:URIEncoding="GBK"。修改后的Connector设置为: 
  
<Connector port="8080" maxHttpHeaderSize="8192" 
               maxThreads="150" minSpareThreads="25" maxSpareThreads="75" 
               enableLookups="false" redirectPort="8443" acceptCount="100" 
               connectionTimeout="20000" disableUploadTimeout="true" URIEncoding="GBK" /> 
  * 注意修改后重新启动tomcat才能起作用。 

4.运行出错时,数据库中的所有数据丢失 
现象: 
   运行出错后,数据库中的所有数据丢失,包括流程定义文件 

原因: 
   jbpm.hibernate.cfg.xml文件中有一个配置 
   <property name="hibernate.hbm2ddl.auto" value="create-drop"/> 

   几个参数的意思,我解释一下: 
   validate    加载hibernate时,验证创建数据库表结构 
   create      每次加载hibernate,重新创建数据库表结构,这就是导致数据库表数据丢失的原因。 
   create-drop 加载hibernate时创建,退出是删除表结构 
   update      加载hibernate自动更新数据库结构 
   none        不进行任何操作 

   由于jbpm4的默认配置为create-drop,所以出现以上问题 

解决方法: 
   将jbpm.hibernate.cfg.xml文件中修改如下 
   <property name="hibernate.hbm2ddl.auto" value="none"/> 

5.在eclipse3.5流程设计器上设计流程时,中文出现乱码 
现象: 
   将流程设计好之后,点击保存,再查看代码,发现中文是乱码 

原因: 
   不清楚,应该是插件的bug 

解决方法: 
   将流程设计好之后,不要点保存,先将界面切换到代码窗口,这时可以看到中文,再点击保存 

6.无法布署zip流程定义文件 
现象: 
   提示以下错误 
   streams type cannot be used in batching 
   2009-11-26 15:58:07 org.hibernate.event.def.AbstractFlushingEventListener performExecutions 
   严重: Could not synchronize database state with session 
   org.hibernate.exception.GenericJDBCException: could not insert: [org.jbpm.pvm.internal.lob.Lob] 

原因: 
   当把数据值增加超过100时,hiberate就出现了这个异常--这意味着Oracle JDBC不允许流操作以批量方式执行 

解决方法: 
   在jbpm.hibernate.cfg.xml文件的<session-factory>下,添加 
     <property name="hibernate.jdbc.batch_size">0</property> 
   即可 

7.无法布署zip流程定义文件 
现象: 
   提示错误 
   Could not synchronize database state with session 
   org.hibernate.exception.GenericJDBCException: could not insert: [org.jbpm.pvm.internal.lob.Lob] 

原因: 
   布署ZIP流程定义文件时hiberate无法插入blob 
   用网络上提议的方法,添加以下配置 
     <property name="hibernate.jdbc.batch_size">0</property> 
     <property name="hibernate.jdbc.use_streams_for_binary">true</property> 
   也不行 

解决方法: 
   采用以上方法,还是无法通过,最后重启操作系统,就解决了,奶奶的。 

8.布署zip流程定义文件成功,但是数据库中中文为乱码 
现象: 
   <?xml version="1.0" encoding="GBK"?> 
   <process name="leave" xmlns="http://jbpm.org/4.0/jpdl"> 
      <start g="201,14,48,48" name="开始"> 
         <transition g="-42,-10" name="请假" to="填写请假单"/> 
      </start> 
      ... 
   由于我的oracle9数据库格式为GBK,所以XML文件头为<?xml version="1.0" encoding="GBK"?> 
   含有中文的流程定义zip文件已经成功保存到了blob字段内,但是中文名为乱码 
   或者提示XML parse error,无法保存到数据库 

原因: 
   repositoryService.createDeployment().addResourcesFromZipInputStream(new ZipInputStream(item.getInputStream())).deploy(); 
   仅上面一句话,就不知道转了多少次编码,经过测试发现,还是编码方式的问题,最后决定将jbpm4.2的源代码复制到项目中来调试。 

解决方法: 
   在网上发现了这篇文章《Java中压缩与解压--中文文件名乱码解决办法》 
   结果问题还是没有解决 
   最后经过测试,将org.jbpm.pvm.internal.repository.DeploymentImpl类中的方法进行修改 
   public NewDeployment addResourcesFromZipInputStream(CnZipInputStream zipInputStream) { 
        try { 
          ZipEntry zipEntry = zipInputStream.getNextEntry(); 
          while(zipEntry!=null) { 
            String entryName = zipEntry.getName(); 
            byte[] bytes = IoUtil.readBytes(zipInputStream); 
            
            //如果是流程定义文件(不是图片),则重新编码,再生成字节数组 
            if(entryName.endsWith(".xml")){ 
                String s=new String(bytes,"utf-8"); 
                bytes=s.getBytes(); 
            } 
                 
            if (bytes!=null) { 
              addResourceFromStreamInput(entryName, new ByteArrayStreamInput(bytes)); 
            } 
            zipEntry = zipInputStream.getNextEntry(); 
          } 
        } catch (Exception e) { 
          throw new JbpmException("couldn't read zip archive", e); 
        } 
        return this; 
      } 
   问题解决,肯定是开始有一步编码方式是用UTF-8,中间你就是再怎么用GBK转码,都不会成功,这里先用UTF-8编码生成字符串,再转一次编码就成功了。 

参考文献 
1.dzq2008. JBPM4项目和tomcat6.0的兼容问题. http://dzq0371.iteye.com/blog/509632 
2.碰到jBPM的编码问题了. http://rednaxelafx.iteye.com/blog/522436 
3.如何解决Tomcat下中文乱码问题. http://www.iteye.com/topic/251743?page=1 
4.关于Hibernate一个配置参数hibernate.hbm2ddl.auto. http://linshiquan.iteye.com/blog/263170
目录
打赏
0
0
0
0
2
分享
相关文章
安装JBpm
原文 http://www.cnblogs.com/default/archive/2012/02/28/2370673.html   自动安装(ant start.demo)太慢,所以手动安装。
895 0
jBPM 5 的点滴
发布地址 http://sourceforge.net/projects/jbpm/files/jBPM%205/  http://www.jboss.org/jbpm/documentation   jBPM5引入了新的API、新的工具并支持BPMN 2.
1036 0
jbpm5.1介绍(7)
Junit测试评估流程 评估流程的界面如下: 这个示例里边用到了Script Task,Service Task和User Task Log执行记录日志的功能,打印出一段信息 RegisterRequest调用外部预定义的程序或者其它引擎的程序 Two调用人工任务 看一下调用的测试程...
837 0
jbpm5.1介绍(2)
快速开始  首先下载jBPM,http://sourceforge.net/projects/jbpm/files/ 可以有选择性的下载: bin:jBPM的二进制文件和它们的依赖包 src:核心模块的源码 gwt-console:jBPM的控制台,包括服务端和客户端 docs:文档 ...
1183 0
jbpm5.1介绍(1)
介绍 jBPM是一个灵活的业务流程管理(BPM)套件。这使得业务分析师和开发人员之间的桥梁。传统的BPM引擎有一个重点,是有限的非技术人员。 jBPM的有两个重点:它提供了一种方式,企业用户和开发人员喜欢它的流程管理功能。
1233 0
jbpm5.1介绍(8)
Junit测试或流程 下面的示例中使用的是或流程,看如下流程图 判断节点的值是大于0,大于10还是大于20 看测试程序 public void testInclusiveSplit() throws Exception { KnowledgeBase kbase = createKnowledgeBase("BPMN2-InclusiveSplit.
749 0
jbpm5.1介绍(10)
Junit测试异常事件触发 下面的示例中测试在程序中触发异常事件的流程,流程如下 测试程序 public void testTimerBoundaryEventInterrupting() throws Exception { KnowledgeBase kbase = crea...
896 0
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等