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 }
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 */
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 */
二.编写自定义类加载器
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 }
使用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>.得到自定义加载器
当你的才华还撑不起你的野心的时候,你就应该静下心来学习。当你的能力还驾驭不了你的目标的时候,你就应该沉下心来历练。问问自己,想要怎样的人生。
欢迎交流学习技术交流,个人微信: "JasonYin2020"(添加时请备注来源及意图备注)
作者: 尹正杰, 博客: https://www.cnblogs.com/yinzhengjie/p/9280564.html