`

【转】JBPM4常用错误汇总

    博客分类:
  • jbpm
阅读更多

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.javaeye.com/blog/509632
2.碰到jBPM的编码问题了. http://rednaxelafx.javaeye.com/blog/522436
3.如何解决Tomcat下中文乱码问题. http://www.javaeye.com/topic/251743?page=1
4.关于Hibernate一个配置参数hibernate.hbm2ddl.auto. http://linshiquan.javaeye.com/blog/263170

本文摘自:http://wangyang0311.javaeye.com/blog/782210

分享到:
评论

相关推荐

    JBPM4 常用表结构及其说明

    ### JBPM4 常用表结构及其说明 JBPM4是JBPM(Java Business Process Model)的一个版本,主要用于提供业务流程管理的功能。它通过一组数据库表来支持工作流的执行与跟踪。本文将详细介绍JBPM4中的常用表结构及其...

    Jbpm4常用操作.doc

    Jbpm4常用操作 Jbpm4是Java流程管理系统(Java Business Process Management System),它提供了一种基于工作流的方式来定义、执行和管理业务流程。下面是Jbpm4常用操作的详细说明: 一、流程定义 流程定义是Jbpm...

    jbpm4jbpm5

    jbpm4jbpm5是关于jbpm流程管理框架的专题,涵盖了jbpm4和jbpm5两个主要版本。jbpm是一个开源的工作流管理系统,用于帮助开发者实现业务流程自动化。以下是基于给定文件的信息,深入解析jbpm4和jbpm5的知识点: 1. *...

    JBPM4运行时错误异常总结

    在使用JBPM4的过程中,可能会遇到各种运行时错误和异常,这些异常通常是由于配置问题、依赖冲突或者代码错误导致的。下面将详细讨论在JBPM4中遇到的一些常见异常及其解决方法。 1. `org.jbpm.api.JbpmException: ...

    jbpm4案例源码

    这个"jbpm4案例源码"包含了jbpm4系统的核心组件和示例代码,帮助开发者深入理解jbpm4的工作原理和实践应用。《jBPM4工作流应用开发指南.pdf》这本书则进一步阐述了如何利用jbpm4来构建工作流应用。 jbpm4的核心功能...

    jBPM4工作流应用开发指南.pdf

    jBPM4作为该系列的一个版本,提供了一套完整的流程管理和执行框架,允许开发者构建灵活的工作流应用,以实现业务流程自动化。本指南由胡奇编写,旨在为开发者提供一个全面的参考,帮助他们快速学习和掌握使用jBPM4...

    jBPM4完整例子

    **jBPM4完整例子详解** jBPM4是一个企业级的工作流管理系统,它提供了强大的业务流程管理和工作流实现能力。这个"jBPM4完整例子"是一个基于SSH(Struts2、Spring、Hibernate)框架集成jBPM4.4的请假流程应用,非常...

    JBPM4 表 详细介绍

    ### JBPM4 表结构详解 JBPM (JBoss Business Process Management) 是一款开源的工作流管理系统,被广泛应用于业务流程管理领域。JBPM4作为其一个版本,具有丰富的功能及特性,尤其在流程管理和任务分配方面表现突出...

    JBPM4 开发文档 实例 流程

    **JBPM4 开发文档详解** JBPM4(Java Business Process Management 4)是一个轻量级的开源工作流管理系统,基于J2EE架构,旨在帮助开发者实现复杂的业务流程自动化。该项目于2004年10月18日发布2.0版本,并随后成为...

    JBPM4 java源代码分包详解JBPM4

    【JBPM4 Java源代码分包详解】 JBPM4是一个流行的开源工作流管理系统,它提供了一整套用于设计、执行和管理业务流程的工具。在深入研究JBPM4的源代码时,我们可以从其包结构中了解到框架的核心功能和设计理念。 1....

    jBPM4工作流示例jbpmDemo

    **jbPM4工作流示例jbpmDemo**是一款基于jBPM4的工作流管理系统实例,旨在帮助用户深入了解和掌握jBPM4工作流引擎的使用方法。jBPM4是一款强大的开源工作流管理系统,它提供了流程建模、执行以及监控等功能,广泛应用...

    jbpm4_table

    jbpm4。4中表之间的关系,jbpm4 表关系

    jbpm4web-JBPM4的一个web版本的Demo

    【jbpm4web - JBPM4的Web版Demo详解】 JBPM4是一个开源的工作流管理系统,它提供了强大的业务流程管理(BPM)功能,包括流程设计、执行、监控和优化。"jbpm4web"是基于JBPM4开发的一个Web版本的示例项目,旨在帮助...

    jBPM 4使用手记

    **jBPM 4 使用手记** jBPM(Java Business Process Management)是一个开源的工作流管理系统,它提供了全面的业务流程管理和工作流解决方案。在jBPM 4这个版本中,开发者可以利用其强大的功能来设计、执行和管理...

    jbpm4 完整的请假流程

    jbpm4是一个基于Java的企业级工作流管理系统,用于设计、执行和管理业务流程。这个“jbpm4 完整的请假流程”是一个示例项目,旨在帮助初学者理解和实践jbpm4在处理业务流程,尤其是请假申请流程中的应用。在这个项目...

    jBPM4入门基础环境配置

    **jBPM4入门基础环境配置** jBPM(Java Business Process Management)是一个开源的工作流管理系统,主要用于处理业务流程的建模、部署、执行和监控。jBPM4是其第四个主要版本,它提供了丰富的功能,包括流程定义、...

    jbpm4服务接口方法汇总

    ### jbpm4服务接口方法汇总 #### Process Engine与Service组件概览 在JBPM框架中,`ProcessEngine`是核心的工厂类,负责提供一系列的服务接口。这些接口为流程管理和控制提供了必要的支持。通过初始化`...

    jbpm4的员工请假例子原码

    【jbpm4的员工请假例子原码】是一个基于jbpm4流程引擎的示例项目,展示了如何在企业中实现一个简单的员工请假流程。这个Web项目是完全开源的,可以通过Eclipse这样的集成开发环境来打开和运行,对于学习和理解jbpm4...

    jbpm4基本jar包

    【jbpm4基本jar包】是企业级工作流管理系统Jboss Business Process Management Suite(简称jbpm)的一个核心组件集合。jbpm4是一个开源的工作流程管理框架,它提供了强大的业务流程建模、执行和监控能力,使得开发...

Global site tag (gtag.js) - Google Analytics