java中jvm怎么卸载_JVM 类的卸载

1、当某个类被加载,连接和初始化后,它的生命周期就开始了。当代表这个类的Class对象不再被引用,即不可触及时,Class对象就会结束生命周期,这个类在方法区内的数据也会被卸载,从而结束这个类的生命周期。

2、一个类何时结束生命周期,取决于代表它的Class对象何时结束生命周期。

3、由Java虚拟机自带的类加载器所加载的类,在虚拟机的生命周期中,始终不会被卸载。前面已经介绍过,Java虚拟机自带的类加载器包括根加载器、扩展类加载器和系统加载器。Java虚拟机会始终引用这些类加载器,而这些类加载器则会始终引用它们所加载的Class对象,因此这些Class对象始终是可触及的。

4、由用户自定义的类加载器所加载的类是可以被卸载的。

5、运行以上程序时,Sample类由loader1加载。在类加载器的内部实现中,用一个Java集合来存放所加载类的引用。另一方面,一个Class对象总是会引用它类加载器,调用Class对象的getClassLoader()方法,就会获得它的类加载器。由此可见,代表Sample类的Class实例与loader1之间为双向关联关系。

一个类的实例总是引用代表这个类的Class对象。在Object类中定义了getClass()方法,这个方法返回代表对象所属的Class对象的引用。此外,所有的Java类都有一个静态属性class,它引用代表这个类的Class对象。

6、类卸载的例子

public class MyTest161 extends ClassLoader{

private String className;

//目录

private String path;

private final String fileExtension = ".class";

public MyTest161(String classLoadName){

super(); //将系统类加载器当做该类加载器的父加载器

this.className = classLoadName;

}

public MyTest161(ClassLoader parent, String classLoadName){

super(parent); //显示指定该类加载器的父加载器器

this.className = classLoadName;

}

public void setPath(String path) {

this.path = path;

}

@Override

public String toString() {

return "[" + this.className + "]";

}

@Override

protected Class> findClass(String clasName) throws ClassNotFoundException {

System.out.println("findClass invoked:" + clasName);

System.out.println("class loader name: " + this.className);

byte[] data = this.loadClassData(clasName);

return this.defineClass(clasName,data, 0, data.length);

}

private byte[] loadClassData(String className){

InputStream is = null;

byte[] data = null;

ByteArrayOutputStream baos = null;

try{

className = className.replace(".","//");

//System.out.println("className:" +this.className);

is = new FileInputStream(new File(this.path + className + this.fileExtension));

baos = new ByteArrayOutputStream();

int ch = 0;

while ( -1 != (ch = is.read())){

baos.write(ch);

}

data = baos.toByteArray();

}catch (Exception ex){

ex.printStackTrace();

}finally {

try {

is.close();

baos.close();

}catch (Exception ex){

ex.printStackTrace();

}

}

return data;

}

public static void main(String[] args) throws Exception{

MyTest161 loader1 = new MyTest161("loader1");

loader1.setPath("D:/temp/");

Class> clazz = loader1.loadClass("com.example.jvm.classloader.MyTest1"); //

System.out.println("class:" + clazz.hashCode());

Object object = clazz.newInstance();

System.out.println(object);

loader1 = null;

clazz = null;

object = null;

System.gc(); // 垃圾回收。实际开发不会用

System.out.println();

loader1 = new MyTest161("loader1");

loader1.setPath("D:/temp/");

clazz = loader1.loadClass("com.example.jvm.classloader.MyTest1"); //

System.out.println("class:" + clazz.hashCode());

object = clazz.newInstance();

System.out.println(object);

System.out.println();

}

}

设置JVM参数 -XX:+TraceClassUnloading

addc7542d45b47e99d0fd2002ddf00b1.png

运行结果:

findClass invoked:com.example.jvm.classloader.MyTest1

class loader name: loader1

class:21685669

com.example.jvm.classloader.MyTest1@7f31245a

findClass invoked:com.example.jvm.classloader.MyTest1

class loader name: loader1

class:1173230247

[Unloading class com.example.jvm.classloader.MyTest1 0x0000000100061028]

com.example.jvm.classloader.MyTest1@330bedb4

7、使用JVisualVM查看类的卸载

增加睡眠1分钟(JVisualVM的使用参考JVisualVM监控本地Java进程)

6b16a94745382aa8cec409de79735045.png

然后使用JVisualVM观察,可以发现已经卸载类的总数为1

9c8e5aadbaa2d1d99ad66c4dc4108d6c.png

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值