浅析类加载

Posted zhh19981104

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了浅析类加载相关的知识,希望对你有一定的参考价值。

前言:总结一下类加载的过程,于静态代码块、静态变量、事例变量等一些构建方式。因为这部分挺绕的,容易出错。

1、java中可以将对象分为两大体系:字节码对象和实例对象

1.1、字节码对象:

每个类在加载(将类读到内存)都会创建一个字节码对象,且这个对象在一个JVM内存中是唯一.此对象中存储的是类的结构信息.所以可以说字节码对象是获取类结构信息的入口.

每个类对应的类字节码文件在jvm中都是唯一的【验证】

 

1     /**
2      * 测试类的字节码文件是否是唯一的
3      */
4     private static void testClassLoader_01(){
5         Class<?> class_1 = Object.class;
6         Class<?> class_2 = Object.class;
7         System.out.println(class_1 == class_2);
8         // 输出结果:true
9     }

 

 

 

 

 

字节码对象的获取方式?(常用方式有三种)

  • 类名.class可以使用Class<?> Class<类名>来接收
  • Class.forName(“包名.类名”)编译阶段无法确定字符串对应的类名是否准确,只能用Class<?>泛型通配符来接收
  • 类的实例对象.getClass()
 1     /**
 2      * 三种类加载方式
 3      */
 4     private static void testClassLoader_02() throws ClassNotFoundException {
 5         Class<?> class_1 = Object.class;
 6         Class<?> class_2 = Class.forName("java.lang.Object");
 7         Class<?> class_3 = new Object().getClass();
 8         System.out.println((class_1 == class_2)&&(class_2 == class_3));
 9         // 注意:三种类加载方式,加载的是同一个类字节码文件
10     }

 

类加载时:

  静态代码块:可以执行,但不一定会执行。

  静类变量:可以初始化,但不一定会初始化。

 

1 class C_Object{
2     static{
3         System.out.println("进入了静态代码块");
4     }
5 }

 

 

 

  

1     private static void testClassLoader_03() throws ClassNotFoundException {
2         // 这个不会输出 进入了静态代码块
3         Class<?> class_1 = Object.class;
4         // 这个会输出  进入了静态代码块
5         Class<?> class_2 = Class.forName("com.turtle.oop.C_Object");
6     }

 

 

 

  

 对Class.forName()进行剖析:

  

1      // arg1:这个是我们要加载的字节码类   
2         // arg2:这个是确定我们在类加载的时候是否执行静态代码块,默认是true,执行,如果为false就不会执行静态代码块
3         // arg3:使用类加载器来完成类的加载
4         Class<?> c3 = Class.forName("com.turtle.oop.ClassObject",true,Thread.currentThread().getContextClassLoader());

 

 

 

 

总结案例:

 

 1 package com.turtle.oop;
 2 
 3 import java.util.HashMap;
 4 import java.util.Map;
 5 
 6 public class Test_ClassLoader_02 {
 7     public static void main(String [] args) throws ClassNotFoundException {
 8         Class<?> class_01 = Class.forName("com.turtle.oop.C_Object_2",true,Thread.currentThread().getContextClassLoader());
 9         // 出错
10     }
11 }
12 class C_Object_2{
13 /*    为类中的静态变量赋值是按顺序下来的
14     1-加载ClassObject_03类到内存中
15     2-为ClassObject_03类型的变量instance赋值,会调用构造方法
16     3-构造方法中使用了静态变量map,但是此时的静态变量map还没有赋值*/
17     static C_Object_2 instance=new C_Object_2();
18     // 如果和上面的交换位置也就不会出错了、如果改为非静态的就不会出错了
19     static Map<String,String> map = new HashMap<>();
20     public C_Object_2(){
21         map.put("Key_01","Value_01");
22     }
23 }

 

 

 报错:空指针错误

 

D:ProgramsJDK8injava.exe "-javaagent:D:ProgramsJetBrainsIntelliJ IDEA 2018.3.5libidea_rt.jar=55074:D:ProgramsJetBrainsIntelliJ IDEA 2018.3.5in" -Dfile.encoding=UTF-8 -classpath D:ProgramsJDK8jrelibcharsets.jar;D:ProgramsJDK8jrelibdeploy.jar;D:ProgramsJDK8jrelibextaccess-bridge-64.jar;D:ProgramsJDK8jrelibextcldrdata.jar;D:ProgramsJDK8jrelibextdnsns.jar;D:ProgramsJDK8jrelibextjaccess.jar;D:ProgramsJDK8jrelibextjfxrt.jar;D:ProgramsJDK8jrelibextlocaledata.jar;D:ProgramsJDK8jrelibext
ashorn.jar;D:ProgramsJDK8jrelibextsunec.jar;D:ProgramsJDK8jrelibextsunjce_provider.jar;D:ProgramsJDK8jrelibextsunmscapi.jar;D:ProgramsJDK8jrelibextsunpkcs11.jar;D:ProgramsJDK8jrelibextzipfs.jar;D:ProgramsJDK8jrelibjavaws.jar;D:ProgramsJDK8jrelibjce.jar;D:ProgramsJDK8jrelibjfr.jar;D:ProgramsJDK8jrelibjfxswt.jar;D:ProgramsJDK8jrelibjsse.jar;D:ProgramsJDK8jrelibmanagement-agent.jar;D:ProgramsJDK8jrelibplugin.jar;D:ProgramsJDK8jrelib
esources.jar;D:ProgramsJDK8jrelib
t.jar;D:ProjectHomeWorks_FrameWork1-JavaEE-Heightenjavaee	argetclasses com.turtle.oop.Test_ClassLoader_02
Exception in thread "main" java.lang.ExceptionInInitializerError
    at java.lang.Class.forName0(Native Method)
    at java.lang.Class.forName(Class.java:348)
    at com.turtle.oop.Test_ClassLoader_02.main(Test_ClassLoader_02.java:8)
Caused by: java.lang.NullPointerException
    at com.turtle.oop.C_Object_2.<init>(Test_ClassLoader_02.java:15)
    at com.turtle.oop.C_Object_2.<clinit>(Test_ClassLoader_02.java:12)
    ... 3 more

Process finished with exit code 1

 

 

 

以上是关于浅析类加载的主要内容,如果未能解决你的问题,请参考以下文章

Android 逆向类加载器 ClassLoader ( 类加载器源码简介 | BaseDexClassLoader | DexClassLoader | PathClassLoader )(代码片段

Android的类加载浅析

Android的类加载浅析

浅析ClassLoader

Android 逆向ART 脱壳 ( DexClassLoader 脱壳 | DexClassLoader 构造函数 | 参考 Dalvik 的 DexClassLoader 类加载流程 )(代码片段

Android 逆向ART 脱壳 ( DexClassLoader 脱壳 | DexClassLoader 构造函数 | 参考 Dalvik 的 DexClassLoader 类加载流程 )(代码片段