java.lang.NoSuchMethodError问题解决

最近做系统版本迭代是时候,报java.lang.NoSuchMethodError(使用spring boot框架)

Exception in thread"main" java.lang.NoSuchMethodError:org.springframework.http.MediaType.getCharset()Ljava/nio/charset/Charset;

   atorg.springframework.web.client.RestTemplate$AcceptHeaderRequestCallback.getSupportedMediaTypes(RestTemplate.java:757)

   atorg.springframework.web.client.RestTemplate$AcceptHeaderRequestCallback.doWithRequest(RestTemplate.java:733)

   atorg.springframework.web.client.RestTemplate$HttpEntityRequestCallback.doWithRequest(RestTemplate.java:795)

   atorg.springframework.web.client.RestTemplate.doExecute(RestTemplate.java:617)

   atorg.springframework.web.client.RestTemplate.execute(RestTemplate.java:580)

   atorg.springframework.web.client.RestTemplate.postForObject(RestTemplate.java:380)

   atcn.com.bsfit.frms.engine.Engine_Test.main(Engine_Test.java:135)

解决方法:依赖包与主包依赖包共用jar时,出现jar环绕依赖,由于jar版本不一致导致此错误。替换jar后解决。

以下是java.lang.NoSuchMethodError通用解决思路:

一般在项目依赖比较复杂或者java运行的环境有问题时同一类型的jar包有不同版本存在,本质上说是JVM找不到某个类的特定方法,也就是说JVM加载了错误版本的类。

出现该问题的情形一般有一下几种:

   1、项目依赖复杂。不使用maven管理项目依赖时更容易出现该问题。

              处理的方法是: 如果使用maven,执行maven dependency:tree 人工排除

   2、运行环境问题。一般Java Web程序都运行在容器中,tomcat等。如果容器中已经存在了某个版本的jar包并已经加载了某些类,而web项目中依赖了不同的版本。

              处理方法:保证使用“干净”的容器运行程序,或者在maven依赖中将容器中已经存在的依赖设置为<scope>provided</scope>

   3、依赖的jar包在不修改namespace的情况下打包了某些他的依赖类。比如:junit 打包了org.hamcrest的一些类。

   4、依赖名称的不同,比如Google Collections 和 Guava,其实是一个东西。

要彻底解决这个问题,首先想到的是不让有冲突的jar包上线。

mvn install 到本地后,执行 mvn jarconflict:check 就可以在web工程中检查所有jar包中的class,如果发现重复就报错。

如果线上已经出现该问题,需要进行定位,解决的方法有以下几种:

1、挂jconsole、jvisualvm、jinfo等工具到启动的java进程,查看jvm的classpath。但这种方法有个局限:如果是web程序运行在tomcat等容器下,容器有自己的classloader结构,会加载web工程目录/WEB-INF/lib/下面所有的jar包,jinfo等工具无能为力

2、如果是web工程,可以将下面的jsp以 class_location.jsp 放到出现问题的web工程下:

[xhtml] view plain copy

  1. <%@ page language="java" contentType="text/html; charset=UTF-8"  
  2.     pageEncoding="UTF-8"%>  
  3. <%@ page import='java.io.*,java.security.CodeSource' %>  
  4. <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">  
  5. <html>  
  6. <head>  
  7. <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">  
  8. <title>get the class location</title>  
  9. </head>  
  10. <body>  
  11. <%  
  12. String className = request.getParameter("className");  
  13. PrintWriter printWriter = new PrintWriter(response.getWriter());  
  14. if(className == null || className.isEmpty()){  
  15.     printWriter.write("className shoud be specified");  
  16. }else{  
  17.     Class<?> clazz = null;  
  18.     try{  
  19.         clazz = Class.forName(className);  
  20.     }catch(Exception e){  
  21.           
  22.     }  
  23.     if(clazz == null){  
  24.         printWriter.println(className + "not found");  
  25.     }else{  
  26.         CodeSource codeSource = clazz.getProtectionDomain().getCodeSource();  
  27.         if(codeSource == null){  
  28.             printWriter.println(className + " location not available");  
  29.         }else{  
  30.             String location = codeSource.getLocation().getPath();  
  31.             printWriter.println(className + " location: " + location);  
  32.         }  
  33.     }  
  34. }  
  35. printWriter.flush();  
  36. %>  
  37. </body>  
  38. </html>  

然后访问 http://你的web程序的地址/class_location.jsp?className=需要检查的类名  检查该类的jar包路径。

例如访问XXX/class_location.jsp?className=net.spy.memcached.MemcachedClient

显示:net.spy.memcached.MemcachedClient location: /data/develop/repository/spy/memcached/2.3.1/memcached-2.3.1.jar

需要注意的是,java.* 下面的所有的类,是无法检测的。

 3、跟踪jvm类加载。在java启动参数中添加 -XX:+TraceClassLoading  -XX:+TraceClassUnloading参数,打印jvm的class loading信息


其他类似报错解决:http://www.cnblogs.com/xiaoMzjm/p/4566672.html


版权声明:本文为u010923921原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
THE END
< <上一篇
下一篇>>