Java基础-类加载机制与自定义类Java类加载器(ClassLoader)

         Java基础-类加载机制与自定义类Java类加载器(ClassLoader)

                                    作者:尹正杰

版权声明:原创作品,谢绝转载!否则将追究法律责任。

 

 

  关于类加载器的概念和分类我就不再废话了,因为我在之前的笔记已经提到过了,详细能看到这篇文章的这些概念应该都烂熟于胸啦!如果你是一名小白的话可以参考我以前的笔记:https://www.cnblogs.com/yinzhengjie/p/9272289.html。本篇博客的重点侧重三点,第一,测试静态代码块;第二,编写自定义类加载器;第三,验证四级类加载机制。

 

一.测试静态代码块

1>.静态代码块的作用

  答:静态代码块的作用说白了跟构造方法的功能类似,他们都可以给类的成员变量进行赋值操作,我们可以简单的理解为静态代码块是为了给类初始化,构造方法是为了给对象初始化。

2>.静态代码块在加载类的必须会被执行吗?

  答:答案是否定的,我们在加载类的时候,可以不指定加载静态代码块。但是,只要你新建一个实例或者是通过反射的方式创建一个实例,依然还是会触发静态代码块的执行。我们在加载类是指定不加载静态代码块说白了只是延时它的加载,因为在实例化的过程中依然会加载!

 1 /*
 2 @author :yinzhengjie
 3 Blog:http://www.cnblogs.com/yinzhengjie/tag/Java%E5%9F%BA%E7%A1%80/
 4 EMAIL:y1053419035@qq.com
 5 */
 6 package cn.org.yinzhengjie.jvm;
 7 
 8 public class Person {
 9     public String name;
10     private int age;
11     public String sex;
12 
13     static {
14         System.out.println("I'm yinzhengjie !!!");
15     }
16 
17     public Person(String name, int age, String sex) {
18         this.name = name;
19         this.age = age;
20         this.sex = sex;
21     }
22 
23     public Person() { }
24 
25     public String getName() {
26         return name;
27     }
28 
29     public void setName(String name) {
30         this.name = name;
31     }
32 
33     public int getAge() {
34         return age;
35     }
36 
37     public void setAge(int age) {
38         this.age = age;
39     }
40 
41     public String getSex() {
42         return sex;
43     }
44 
45     public void setSex(String sex) {
46         this.sex = sex;
47     }
48 
49     @Override
50     public String toString() {
51         return "Person{" + "name='" + name + '\'' + ", age=" + age + ", sex='" + sex + '\'' + '}';
52     }
53 }
Person.java 文件内容
 1 /*
 2 @author :yinzhengjie
 3 Blog:http://www.cnblogs.com/yinzhengjie/tag/Java%E5%9F%BA%E7%A1%80/
 4 EMAIL:y1053419035@qq.com
 5 */
 6 
 7 package cn.org.yinzhengjie.jvm;
 8 
 9 import org.junit.Test;
10 
11 import java.lang.reflect.Field;
12 
13 public class TestJVM {
14 
15     @Test
16     public void testClassLoader() throws Exception {
17         //通过ClassLoader.getSystemClassLoader()获取系统加载类,将类的初始化操作设置为false.
18         Class c1 = Class.forName("cn.org.yinzhengjie.jvm.Person",false,ClassLoader.getSystemClassLoader());
19         Field name = c1.getDeclaredField("name");
20         System.out.println(name);
21     }
22 }
23 
24 /*
25 以上代码输出结果如下:
26 public java.lang.String cn.org.yinzhengjie.jvm.Person.name
27 */
TestJVM.java 文件内容(不指定加载静态代码块)
 1 /*
 2 @author :yinzhengjie
 3 Blog:http://www.cnblogs.com/yinzhengjie/tag/Java%E5%9F%BA%E7%A1%80/
 4 EMAIL:y1053419035@qq.com
 5 */
 6 
 7 package cn.org.yinzhengjie.jvm;
 8 
 9 import org.junit.Test;
10 
11 import java.lang.reflect.Field;
12 
13 public class TestJVM {
14 
15     @Test
16     public void testClassLoader() throws Exception {
17         //通过ClassLoader.getSystemClassLoader()获取系统加载类,将类的初始化操作设置为false.
18         Class c1 = Class.forName("cn.org.yinzhengjie.jvm.Person",false,ClassLoader.getSystemClassLoader());
19         Field field = c1.getDeclaredField("name");
20         System.out.println(field);
21         //虽然上面的代码再加载类的过程中设置禁用静态代码块的加载,通过反射获取到的时候依然会触发静态代码块的执行!
22         Object obj = c1.newInstance();
23         field.set(obj,"尹正杰");
24         System.out.println(obj);
25     }
26 }
27 
28 /*
29 以上代码输出结果如下:
30 public java.lang.String cn.org.yinzhengjie.jvm.Person.name
31 I'm yinzhengjie !!!
32 Person{name='尹正杰', age=0, sex='null'}
33  */
TestJVM.java 文件内容(反射触发静态代码块的执行)

 

二.编写自定义类加载器

1>.编写自定义类加载器

 1 /*
 2 @author :yinzhengjie
 3 Blog:http://www.cnblogs.com/yinzhengjie/tag/Java%E5%9F%BA%E7%A1%80/
 4 EMAIL:y1053419035@qq.com
 5 */
 6 package cn.org.yinzhengjie.jvm;
 7 
 8 import java.io.ByteArrayOutputStream;
 9 import java.io.FileInputStream;
10 
11 /**
12  * 自定义ClassLoader
13  */
14 public class MyClassLoader extends ClassLoader {
15     protected Class<?> findClass(String name) {
16         try {
17             String path = "D:\\BigData\\JavaSE\\yinzhengjieData\\" + name + ".class" ;
18             FileInputStream in = new FileInputStream(path) ;
19             ByteArrayOutputStream baos = new ByteArrayOutputStream() ;
20             byte[] buf = new byte[1024] ;
21             int len = -1 ;
22             while((len = in.read(buf)) != -1){
23                 baos.write(buf , 0 , len);
24             }
25             in.close();
26             byte[] classBytes = baos.toByteArray();
27             return defineClass(classBytes , 0 , classBytes.length) ;
28         } catch (Exception e) {
29             e.printStackTrace();
30         }
31         return null ;
32     }
33 }

2>.调用自定义类加载器

  在“D:\BigData\JavaSE\yinzhengjieData”目录下编写测试代码如下:

 1 /*
 2 @author :yinzhengjie
 3 Blog:http://www.cnblogs.com/yinzhengjie/tag/Java%E5%9F%BA%E7%A1%80/
 4 EMAIL:y1053419035@qq.com
 5 */
 6 
 7 public class Yinzhengjie {
 8     private String name;
 9     private int age;
10     static {
11         System.out.println("I'm yinzhengjie !");
12     }
13 
14     public Yinzhengjie(String name, int age) {
15         this.name = name;
16         this.age = age;
17     }
18 
19     public Yinzhengjie() {
20     }
21 
22     public String getName() {
23         return name;
24     }
25 
26     public void setName(String name) {
27         this.name = name;
28     }
29 
30     public int getAge() {
31         return age;
32     }
33 
34     public void setAge(int age) {
35         this.age = age;
36     }
37 
38     @Override
39     public String toString() {
40         return "Yinzhengjie{" + "Name = " + name + ", Age = " + age + '}';
41     }
42 }
Yinzhengjie.java 文件内容

  使用javac命令编译Java代码。

  调用咱们自定义的类加载器,测试代码如下:

 1 /*
 2 @author :yinzhengjie
 3 Blog:http://www.cnblogs.com/yinzhengjie/tag/Java%E5%9F%BA%E7%A1%80/
 4 EMAIL:y1053419035@qq.com
 5 */
 6 package cn.org.yinzhengjie.jvm;
 7 
 8 import java.lang.reflect.Constructor;
 9 
10 public class MyClassLoaderApp {
11     public static void main(String[] args) {
12         //创建自定义类加载器实例
13         ClassLoader cl = new MyClassLoader() ;
14         cl.getParent();
15         try {
16             Class clz = cl.loadClass("Yinzhengjie") ;
17             Constructor constructor = clz.getConstructor(String.class, int.class);
18             Object obj = constructor.newInstance("尹正杰", 18);
19             System.out.println(obj);
20         } catch (Exception e) {
21             e.printStackTrace();
22         }
23     }
24 }
25 
26 /*
27 以上代码执行结果如下:
28 I'm yinzhengjie !
29 Yinzhengjie{Name = 尹正杰, Age = 18}
30  */

 

三.验证四级类加载机制

1>.根类加载器是得不到的

 

2>.得到扩展类加载器

 

3>.得到系统类加载器

 

4>.得到自定义加载器

 

posted @ 2018-07-08 17:04  尹正杰  阅读(3009)  评论(0编辑  收藏  举报