快精灵印艺坊 您身边的文印专家
广州名片 深圳名片 会员卡 贵宾卡 印刷 设计教程
产品展示 在线订购 会员中心 产品模板 设计指南 在线编辑
 首页 名片设计   CorelDRAW   Illustrator   AuotoCAD   Painter   其他软件   Photoshop   Fireworks   Flash  

 » 彩色名片
 » PVC卡
 » 彩色磁性卡
 » 彩页/画册
 » 个性印务
 » 彩色不干胶
 » 明信片
   » 明信片
   » 彩色书签
   » 门挂
 » 其他产品与服务
   » 创业锦囊
   » 办公用品
     » 信封、信纸
     » 便签纸、斜面纸砖
     » 无碳复印纸
   » 海报
   » 大篇幅印刷
     » KT板
     » 海报
     » 横幅

对于不同级别classloader define的类的问题处理


问题描述:程序代码中,执行下列语句:Object mapperObj = Class.forName(mapperClassName).newInstance();MapperInterface mapper = (MapperInterface)mapperObj; 报ClassCastException。

bug fix:  1.分别取得运行环境下mapperObj和MapperInerface.class的 classloader:   mapperObj.getClass().getClassLoader() : sun.misc.Launcher$AppClassLoader    MapperInterface.class.getClass().getClassLoader() : WebContainerClassLoader2.由jvm classload 机制可知,jvm load class 分四个层次:  第一层为bootstrapclassloader : 主要负责load rt.jar等jvm必须的jar包中的类。  第二层为extclassloader : 主要负责load 被置于java.ext.dirs属性值所指路径(默认%JAVA_HOME%/lib/ext) 中所有的class.其实现类为sun.misc.Launcher$ExtClassLoader   第三层为systemclassloader : 负责load 被置于CLASSPATH路径中的类。 其实现类一般为sun.misc.Launcher$AppClassLoader   第四层为appclassloader : 由应用程序设计者继续ClassLoader并实现完成相应user-defined ClassLoader。用于根据应用程序需要加载并不是设计时就知道的类。

具体的load策略偶就不多写了,很多文章上都有,总之两句话: 当define一个类的时候,低层classloader会向上层询问是否已经define,有则直接拿来用;当load一个类的时候,同样低层向高层询问是否能find到,能就直接拿来用。

由此可知,由于原有系统原因, mapperObj被 systemclassloader define;而这里使用的接口是被appclassloader WebContainerClassLoader define的。所以会造成 ClassCastException错误。用instanceof也可发现mapperObj 确实不是 MapperInterface的实例。

由于原系统原因,无法通过改动其他代码完成更换mapperObj classload的动作,而运行到当前代码时,mapperObj 已经被define,所以无法通过forName方式的参数更改其class loader,后面的代码就无法调用其方式。

解决方式:采用类反射,换有Object定义的属性接 mapperObj;在下面的代码中,利用mapperObj.getClass().getInterface()方式判定是否其继续了 MapperInterface。 然后用反射调用其方式。例:

mapper = Class.forName(mapperClassName).newInstance();

Class[] tmpInterface = mapper.getClass().getInterfaces();   for(int i=0 ;i<tmpInterface.length ;i++){ if(tmpInterface[i].getName().equals("MapperInterface.class.getName()"))    flag = true;}

if(flag){  Class[] tc = new Class[2] ;  tc[0] = String.class;  tc[1] = HttpServletRequest.class;  Method mapperFunc = mapper.getClass().getMethod( "mapFunction", tc );  Object res = mapperFunc.invoke( mapper, new Object[] { event.getServletClassName(),req } );}






返回类别: 教程
上一教程: 递归函数之JAVA演绎
下一教程: struts为每个独立哀求创建一个Action的方式

您可以阅读与"对于不同级别classloader define的类的问题处理"相关的教程:
· 关于JAVAMAIL处理中文附件的问题
· java 与 mysql 中文问题的处理
· JAVA程序开发小经验 - 使用ObjectStream会出现的问题
· 在java开发过程中常常遇到数据类型的问题
· 每个初学JAVA者都应该搞懂的问题
    微笑服务 优质保证 索取样品