定位web开发中的jar包冲突
几天前接到一个技术支持,现场的描述是war从tomcat移植到was7后部分页面不能编译,从页面和日志看,错误很明确,提示找不到方法“noSuchMethod”,可以肯定的是应用服务器加载了错误版本的class。
处理的过程:
1.从现场要来对应的war,找到jar,反编译发现class中存在相关的方法
2.可能lib中有多个同名的class,使用winrar的搜索功能,搜索整个lib下的所有jar,发现两个,但是反编译看class是正确的。
3.可能was7加载了应用服务器lib目录下或共享目录的jar,到网上搜了下was7的类加载次序,受was的设置和共享库等因素的影响,找起来不就麻烦,想想可以写一个jsp找出指定class的加载路径,以后遇到类似的问题还可以复用。
完整的jsp代码如下:
<%@page pageEncoding="UTF-8"%> <%@page import="java.net.URL" %> <%@page import="java.io.*" %> <%! String getClassPath(String className){ StringBuffer sb=new StringBuffer("<b>className</b>:"+className+"<br>"); try { Class z=Class.forName(className); String path = z.getName().replace(".", "/"); path ="/"+path+".class"; URL url=z.getResource(path); if(url!=null){ sb.append("<b>class loader</b>:"+z.getClassLoader()); sb.append("<br><b>class path</b>:"+url.getPath()); }else{ sb.append("class not loaded by any classLoader"); } } catch (Exception e) { e.printStackTrace(); StringWriter writer = new StringWriter(); PrintWriter pw = new PrintWriter(writer); e.printStackTrace(pw); pw.close(); sb.append("error:"+e); sb.append("<br><pre>"+writer.toString()+"</pre>"); } return sb.toString(); } %> <html> <head><title>find class path</title> <style> *{font:14px;} b{font-weight:bold;} </style> </head> <body> <form action="<%=request.getRequestURI()%>"> class full name:<input type=text name=className size=60 /><br> <input type=submit value=submit /> </form> <hr> <% String className=request.getParameter("className"); if(className!=null && !"".equals(className)){ out.println(getClassPath(className)); } %> </body> </html>
将jsp放到应用中,访问此jsp,在input中输入要定位加载路径的class的全名,测试一下com.ibm.db2.jcc.DB2Driver
能看到加载的class loader和加载的path,输入错误日志中提示的类,发现加载的classpath就在app的lib下面,最后找出实际运行的class和发给我的不是同一个。