几天前接到一个技术支持,现场的描述是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和发给我的不是同一个。
?
?