Java学习者论坛

 找回密码
 立即注册

QQ登录

只需一步,快速开始

手机号码,快捷登录

恭喜Java学习者论坛(https://www.javaxxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,购买链接:点击进入购买VIP会员
JAVA高级面试进阶视频教程Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程

Go语言视频零基础入门到精通

Java架构师3期(课件+源码)

Java开发全终端实战租房项目视频教程

SpringBoot2.X入门到高级使用教程

大数据培训第六期全套视频教程

深度学习(CNN RNN GAN)算法原理

Java亿级流量电商系统视频教程

互联网架构师视频教程

年薪50万Spark2.0从入门到精通

年薪50万!人工智能学习路线教程

年薪50万!大数据从入门到精通学习路线年薪50万!机器学习入门到精通视频教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程 MySQL入门到精通教程
查看: 1584|回复: 0

[默认分类] java.lang.NoSuchMethodError问题处理

[复制链接]
  • TA的每日心情
    开心
    2021-12-13 21:45
  • 签到天数: 15 天

    [LV.4]偶尔看看III

    发表于 2018-7-6 16:19:47 | 显示全部楼层 |阅读模式

       
    一般在项目依赖比较复杂或者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包上线。
       
    这里有我写的一个简单的工具:
    https://code.google.com/p/jarconflict/
       
    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工程下:
       
    <%@ page language="java" contentType="text/HTML; charset=UTF-8"
        pageEncoding="UTF-8"%>
    <%@ page import="java.io.*,java.security.CodeSource" %>
    <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
    <html>
    <head>
    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
    <title>get the class location</title>
    </head>
    <body>
    <%
    String className = request.getParameter("className");
    PrintWriter printWriter = new PrintWriter(response.getWriter());
    if(className == null || className.isEmpty()){
            printWriter.write("className shoud be specified");
    }else{
            Class<?> clazz = null;
            try{
                    clazz = Class.forName(className);
            }catch(Exception e){
                   
            }
            if(clazz == null){
                    printWriter.println(className + "not found");
            }else{
                    CodeSource codeSource = clazz.getProtectionDomain().getCodeSource();
                    if(codeSource == null){
                            printWriter.println(className + " location not available");
                    }else{
                            String location = codeSource.getLocation().getPath();
                            printWriter.println(className + " location: " + location);
                    }
            }
    }
    printWriter.flush();
    %>
    </body>
    </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信息


       
       
      -EOF-
       
       
       
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|Java学习者论坛 ( 声明:本站资料整理自互联网,用于Java学习者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2024-5-22 13:06 , Processed in 0.361048 second(s), 54 queries .

    Powered by Discuz! X3.4

    © 2001-2017 Comsenz Inc.

    快速回复 返回顶部 返回列表