200分给予解答:如何实现JVM类重新加载?

yczz 2010-03-14 01:43:08
去年有一款产品“jrebel”可以实现运行时的类重新加载(包括方法的新增和删除),大家都知道JVM使用Application ClassLoader来加载放在classpath下的类,而且只加载一次下次在访问就会从缓存中取,所以要实现类重载只能是重新创建一个新的ClassLoader在来加载变化后的class。
问题:
1、JVM是在什么时候创建类的ClassLoader
2、创建自己的ClassLoader没什么问题,但是创建后的ClassLoader如何注册到JVM中呢?

不知道有没有JVM的高手可以解答这个问题

注意:此贴并是不讨论如何创建自定义的ClassLoader,而是需要一个JVM的钩子能够实现的需要时自动创建ClassLoader来达到类的重新加载的功能。
...全文
1230 45 打赏 收藏 转发到动态 举报
写回复
用AI写文章
45 条回复
切换为时间正序
请发表友善的回复…
发表回复
vosnow 2011-10-21
  • 打赏
  • 举报
回复
去年的贴子。。。。。。
vosnow 2011-10-21
  • 打赏
  • 举报
回复
ClassLoader只有加载类的功能,没有卸载类的功能!而一个ClassLoader对同一个类只能加载一次,除非是不同的ClassLoader(类或实例)。

换句话说即便你能做到:把自己定义的ClassLoader注册到了JVM中,你的程序从一开始用的ClassLoader都是你自己定义的,你也无法更新已经Load了的Class!

ClassLoader与Class是一个整体,你要想更新Class那你就得创建一个新的ClassLoader实例来Load你更新之后的Class。

所以你的思路有点歪,你要做的不是取代系统默认的ClassLoader,而是怎样管理你自己定义的ClassLoaders
yczz 2010-03-20
  • 打赏
  • 举报
回复
jrebel是去年获得jolt生产力大奖的产品,老外开发的本能还没这个技术,不过是对这个产品实现原理很感兴趣提出来大家一起研究下,如若不懂没关系支持顶下就可以了。
zxcv88888 2010-03-19
  • 打赏
  • 举报
回复
比较奇特的是jrebel ,一直在强调这个
感觉楼在推销这个啊
yczz 2010-03-19
  • 打赏
  • 举报
回复
如果能够随意用自己 classloader加载的话就不用这么麻烦了,因为要实现以插件形式类的动态加载就不能限制开发人员只能使用规定的自定义类加载器来加载class,开发人员可以随意的以new、反射灯方式来加创建对象。
jxygjzh04 2010-03-18
  • 打赏
  • 举报
回复
可以实现运行时的类重新加载(包括方法的新增和删除)

其中——包括方法的新增和删除,我说的实现起来可能就麻烦...期待高人回答。
jxygjzh04 2010-03-18
  • 打赏
  • 举报
回复
虽然我对这方面也是一知半解的,但是我也稍微说一下我的看法

1.首先要了解,jvm启动时候,会形成三个类加载器,这个前面有人说过了:
bootstrap classloader , extension classloader , system classloader

2.而且要知道,classloader 有一种全盘负责委托机制,也就是说,cl要载入一个class的时候,这个class所依赖的所有class都要由这个classloader载入,除非是你显示的让另一个cl载入。而且委托机制是先让父加载器寻找,要是父加载器找不到的时候,在从自己的类路径中寻找。

这样的话,要是想动态载入class的话,是不是可以这样处理:
比如我们要动态加载一个class,名字为 A ,我们可以构造一个A的父类,叫AParent,然后把AParent放在CLASSPATH下(system classloader加载的类路径),父类由system classloader载入,当我们用自己的classloader动态载入A的时候,发现他有个爹...在载入他之前,jvm会先载入他爹..要由system classloader载入。根据多态,调用AParent方法的时候,其实是调用的A的方法。


说的可能不对,大家带着批判的眼光来看,期待高手的解答,别让我脑子里面记得东西都是错误的东西。
jianghg2009 2010-03-18
  • 打赏
  • 举报
回复
学习了,类加载这个东东正想研究研究。。。。。。
yczz 2010-03-17
  • 打赏
  • 举报
回复
jAgent transform,只是在每个类初次加载进JVM的时候调用一次以后Class就会缓存起来,下次再次new这个对象不会在调用这个方法。这也是苦苦找不到注入点的原因。
problc 2010-03-17
  • 打赏
  • 举报
回复
反编译了一下jrebel的代码,里面有两个类TransformNew,AgentInstall
AgentInstall利用Instrumentation进行拦截,TransformNew把自己new出来的对象就行修改,实际new出来的对象已经是jrebel可以控制的对象了.jrebel用的是bytecode的修改
yczz 2010-03-16
  • 打赏
  • 举报
回复
自己构建classloader很多人都可以做,核心问题是你做的classloader如何注册进去,如果说自己写个框架加载自己的类这个目前有很多实现可以参考。比较奇特的是jrebel在使用javaagent注册进去后你程序代码里面不管是用new,还是用反射接口等操作他都能自动检测到并重新加载,这个是我很想知道的。
yczz 2010-03-16
  • 打赏
  • 举报
回复
jiewo
JVM有个插件可以帮助你实现类的重加载~ 是什么插件???

有没有人研究过jrebel,可以谈谈心得
abc130314 2010-03-16
  • 打赏
  • 举报
回复
忘了一点。
如果不想一个一个的用 静态方法实例化。
可以 看看 Thread.setContextClassLoader。
这时候可以用 new.
不过必须得对 线程控制好。
abc130314 2010-03-16
  • 打赏
  • 举报
回复
没有办法用 new ,应该只能用 静态方法实例化。
而且一般的加载方式是,在默认classloader里面有一个接口或者父类。
然后用自定义的加载器加载 要热加载的类。然后把 这个类 实例化转换为 父类或者接口。

具体的应该是,有一个静态方法,这个方法返回一个 默认classloader 已经加载了 的接口或父类 的实例。
然后静态方法首先检查 文件是否已经更新,已经更新就生成一个新的 classloader 加载这个 新的class,然后返回新的实例。

如果 默认classloader 里面没有接口或者父类,那就只能返回Object,然后用反射调用具体的方法。这时候用脚本编写比较方便。


problc 2010-03-16
  • 打赏
  • 举报
回复
[Quote=引用 28 楼 yczz 的回复:]
自己构建classloader很多人都可以做,核心问题是你做的classloader如何注册进去,如果说自己写个框架加载自己的类这个目前有很多实现可以参考。比较奇特的是jrebel在使用javaagent注册进去后你程序代码里面不管是用new,还是用反射接口等操作他都能自动检测到并重新加载,这个是我很想知道的。
[/Quote]
http://java.sun.com/javase/6/docs/jdk/api/attach/spec/com/sun/tools/attach/VirtualMachine.html
http://blog.csdn.net/ThirstyCrow/archive/2008/10/30/3185018.aspx
可以搜一下这个VirtualMachine,Instrumentation,很强大,可以attach到指定的虚拟机上,指定agent,在agent里面进行实现拦截器,重新加载。jdk自带的tool.jar里面的类。
现在很多Java框架如Spring都实现有拦截器,不管是基于动态代理还是Asm那样的直接修改字节码。
zhoujigang 2010-03-16
  • 打赏
  • 举报
回复
顶楼上的,学习中
Rock姜 2010-03-16
  • 打赏
  • 举报
回复
我认为,你可以参考下Tomcat的源码,首先Tomcat编译的时候是通过吧jsp页面动态编译成servlet。然后把编译后的.class文件以文件流的形式传入继承了java.lang.ClassLoader类中重写的defineClass方法中。

既然Tomcat可以做到。那么我们就可以节省jsp页面编译成servlet的过程。
我们通过java中的IO类动态生成.java文件。然后通过Tomcat中的动态编译和加载功能进行加载。

没有试过,但是最近在看Tomcat的源码,都是java写的。
一定可行。
JamesLiu 2010-03-16
  • 打赏
  • 举报
回复
路过,学习了
taolei 2010-03-16
  • 打赏
  • 举报
回复
类重新加载通常有两种方式。javaagent方式和重载classloader方式
jrebel的方式是用java -javaagent参数指定的,类加载不是在classloader上做的。
在java1.5以后,增加了java -javaagent参数,提供了一个动态替换class的方式。
以jrebel为例:运行参数为java -javaagent jrebel.jar
在jrebel.jar里的META-INF目录下的MANIFEST.MF文件中定义Premain-Class
Premain-Class: com.zeroturnaround.javarebel.java5.AgentInstall

该class包含一个 public static void premain(String agentArgs, Instrumentation inst)
的方法作为入口。
在java.lang.instrument.Instrumentation中提供具体的动态替换功能。
当然,要做到动态替换class,还要有class更新监控的功能。
相关知识在API文档中Package java.lang.instrument 及其子目录下有详细的介绍(JDK1.5以后)。


d1984 2010-03-16
  • 打赏
  • 举报
回复
不明白 帮顶了 等高手来 我学习
加载更多回复(24)
面试题包含了不同技术层面的面试问题,同时也能对一些没有面试开发经验的小白给予不可估量的包装, 让你的薪水绝对翻倍, 本人亲试有效.Java面试题84集、java面试专属及面试必问课程,所有的面试题有视屏讲解, 解答方案.以下是部目录: java面试题01.面试的整体流程.mp4 │ Java面试题02.java的垮平台原理.mp4 │ Java面试题03.搭建一个java的开发环境.mp4 │ Java面试题04.java中int占几个字节.mp4 │ Java面试题05.java面向对象的特征.mp4 │ Java面试题06.装箱和拆箱.mp4 │ Java面试题07.==和equals的区别.mp4 │ Java面试题08.String.mp4 │ Java面试题09.讲一下java中的集合.mp4 │ Java面试题10.ArrayList LinkedList.mp4 │ Java面试题11.HashMap和HashTable的区别.mp4 │ Java面试题12.实现一个拷贝文件的使用字节流还是字符串.mp4 │ Java面试题13.线程的实现方式 怎么启动线程怎么区线程.mp4 │ Java面试题14.线程并发库和线程池的作用?.mp4 │ Java面试题15.设计模式和常用的设计模式.mp4 │ Java面试题16.http get post请求的区别.mp4 │ Java面试题17.说说你对Servlet的理解.mp4 │ Java面试题18.Servlet的生命周期.mp4 │ Java面试题19.forward和redirect的区别.mp4 │ Java面试题20.jsp和Servlet的相同点和不同点?.mp4 │ Java面试题21.内置对象和四大作用域和页面传值.mp4 │ Java面试题22.Session和Cookie的区别.mp4 │ Java面试题23.mvc模式和mvc各部实现.mp4 │ Java面试题24.数据库和常用数据库.mp4 │ Java面试题25.关系型数据库的三范式.mp4 │ Java面试题26.事务的四大特征.mp4 │ Java面试题27.mysql数据库最大连接数.mp4 │ Java面试题28.mysql和oracle的页语句(着重说思路).mp4 │ Java面试题29.触发器的使用场景.mp4 │ Java面试题30.存储过程的优点.mp4 │ Java面试题31.jdbc调用存储过程.mp4 │ Java面试题32.简单说一下你对jdbc的理解.mp4 │ Java面试题33.写一个jdbc的访问oracle的列子.mp4 │ Java面试题34.jdbc中preparedStatement比Statement的好处.mp4 │ Java面试题35.数据库连接池的作用.mp4 │ Java面试题36.HTML.mp4 │ Java面试题37.简单介绍了一下Ajax.mp4 │ Java面试题38.js和JQuery的关系.mp4 │ Java面试题39.jQuery中的常用选择器.mp4 │ Java面试题40.jQuery中页面加载完毕事件.mp4 │ Java面试题41.jQuery中Ajax和原生js实现Ajax的关系.mp4 │ Java面试题42.简单说一下html5.mp4 │ Java面试题43.简单说一下css3.mp4 │ Java面试题44.bootstrap的是什么.mp4 │ Java面试题45.什么是框架.mp4 │ Java面试题46.简单介绍一下MVC模式.mp4 │ Java面试题47.简单说一下对mvc框架的理解.mp4 │ Java面试题48.struts2的执行流程或者struts2的原理.mp4 │ Java面试题49.Struts2的拦截器是什么?你都用它干什么?.mp4 │ Java面试题50.Spring MVC的执行流程.mp4 │ Java面试题51.SpringMVC和Struts2的不同.mp4 │ Java面试题52.简单介绍一下Spring或者Spring的两大核心.mp4 │ Java面试题53.AOP是什么?都用它做什么?.mp4 │ Java面试题54.Spring事务的传播特性和隔离级别.mp4 │ Java面试题55.ORM是什么?ORM框架是什么?.mp4 │ Java面试题56.ibatis和hibernate有什么不同.mp4 │ Java面试题57.hibernate对象状态及其转换.mp4 │ Java面试题58:hibernate的缓存.mp4 │ Java面试题59.webservice的使用场景.mp4 │ Java面试题60.Activiti的简单介绍.mp4 │ Java面试题61.linux的使用场景.mp4 │ Java面试题62.linux常用命令.mp4 │ Java面试题63:怎么操作linux服务器.mp4 │ Java面试题64:有没有使用过云主机.mp4 │ Java面试题65:数据库优化方面的事情.mp4 │ Java面试题66:如果查询和定位慢查询.mp4 │ Java面试题67:数据库优化之数据库表设计遵循范式.mp4 │ Java面试题68:选择合适的数据库引擎.mp4 │ Java面试题69:选择合适的索引.mp4 │ Java面试题70:使用索引的一些技巧.mp4 │ Java面试题71:数据库优化之表.mp4 │ Java面试题72:数据库的读写离.mp4 │ Java面试题73:数据库优化之缓存.mp4 │ Java面试题74:sql语句优化小技巧.mp4 │ Java面试题75:批量插入几百万条数据.mp4 │ Java面试题76:有没有使用过redis.mp4 │ Java面试题77:redis的使用场景.mp4 │ Java面试题78:redis存储对象的方式.mp4 │ Java面试题79:redis数据淘汰机制.mp4 │ Java面试题80:java访问redis级redis集群?.mp4 │ Java面试题81:微信公众号和微信开发原理.mp4 │ Java面试题82:怎么把微信和业务平台进行绑定.mp4 │ Java面试题83:项目的和项目参与者.mp4 │ Java面试题84:项目流程和业务注意事项.mp4 │ 面试必问-Mysql索引背后的故事 │ ├─java面试专属 │ ├─1.面试必考之HashMap源码析与实现 │ │ 1.面试必考之HashMap源码析与实现.mp4 │ │ │ ├─2.探索JVM底层奥秘ClassLoader源码析与案例讲解 │ │ 2.探索JVM底层奥秘ClassLoader源码析与案例讲解.wmv │ │ │ ├─3.锁、布式锁、无锁实战全局性ID-悟空 │ │ 3.锁、布式锁、无锁实战全局性ID-悟空.mp4 │ │ │ ├─4.SpringMvc深入理解源码析 │ │ 4.SpringMvc深入理解源码析-悟空.mp4 │ │ │ ├─5.Nosql Redis Jedis常用命令 │ │ 5.Nosql Redis Jedis常用命令-悟空.mp4 │ │ │ ├─6.互联网系统垂直架构之Session解决方案 │ │ 6.互联网系统垂直架构之Session解决方案.mp4 │ │ │ ├─7.布式框架ZooKeeper之服务注册与订阅 │ │ 7.布式框架Zookeeper之服务注册与订阅.mp4 │ │ │ ├─8.高性能网络编程必备技能之IO与NIO阻塞析 │ │ 8.高性能网络编程必备技能之IO与NIO阻塞析.mp4 │ │ │ ├─9.JAVA并发编程之多线程并发同步业务场景与解决方案 │ │ 9.JAVA并发编程之多线程并发同步业务场景与解决方案.wmv │ │ │ ├─10.微服务架构之Spring Cloud Eureka 场景析与实战 │ │ 10.微服务架构之Spring Cloud Eureka 场景析与实战.wmv │ │ │ ├─11.高性能必学之Mysql主从架构实践 │ │ 11.高性能必学之Mysql主从架构实践.mp4 │ │ │ ├─12.架构师不得不知道的Spring事物不能回滚的深层次原因 │ │ 12.架构师不得不知道的Spring事物不能回滚的深层次原因.mp4 │ │ │ ├─13.RPC底层通讯原理之Netty线程模型源码析 │ │ 13.RPC底层通讯原理之Netty线程模型源码析.wmv │ │ │ ├─14.表之后布式下如何保证ID全局唯一性 │ │ 14.表之后布式下如何保证ID全局唯一性.mp4 │ │ │ └─15.大型公司面试必答之数据结构与算法精讲 │ 大型公司面试必答之数据结构与算法(一)-达摩老师.mp4 │ 大型公司面试必答之数据结构与算法(二).mp4 │ ├─面试必问-JVM性能调优 │ JVM性能调优 2018-10-25.mp4 │ ├─面试必问-mybaits源码析 │ │ 鲁班学院-上课笔记mybaits源码析9-05.docx │ │ │ └─mybaits源码析 │ mybaits源码析.mp4 │ ├─面试必问-springcloud架构微服务项目 │ springcloud架构微服务项目.mp4 │ ├─面试必问-SpringMVC源码析 │ SpringMVC源码析.mp4 │ ├─面试必问-webservice原理析 │ webservice原理析.mp4 │ ├─面试必问-使用Springboot快速搭建SSM框架 │ 使用SpringBoot快速搭建SSM框架.mp4 │ ├─面试必问-双十一系统架构之Mysql索引技术剖析 │ 双十一系统架构之Mysql索引技术剖析.mp4 │ ├─面试必问-大牛带你手写dubbo框架 │ 大牛带你手写Dubbo框架.mp4 │ ├─面试必问-实战布式之手写布式事务框架 │ 实战布式之手写布式事务框架.mp4 │ ├─面试必问-带你精通springAOP—面试无忧虑 │ 带你精通AOP——面试无忧虑.mp4 │ ├─面试必问-微服务架构深入浅出讲解springcloud │ 微服务架构 --深入浅出讲解springcloud.mp4 │ ├─面试必问-教你手写MyBatis框架 │ 一小时教你手写MyBatis框架.mp4 │ ├─面试必问-架构杀手锏——java混乱的日志体系 │ java混亂日志体系源码揭秘.mp4 │ ├─面试必问-深入微服务之SpringBoot&Docker1 │ 深入微服务之SpringBoot&Docker.mp4 │ └─面试必问-聊聊哈希算法与HashMap

62,614

社区成员

发帖
与我相关
我的任务
社区描述
Java 2 Standard Edition
社区管理员
  • Java SE
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧