面试官: 谈谈你对Java反射的理解

Java的类加载

Java在真正需要使用一个类时才会去加载类,而不是在启动程序时就载入所有的类,因为大多数使用者都只使用到程序的部分资源,在需要某些功能时再载入某些资源,可以让系统资源运用的更高效。

类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在Jvm的方法区内,然后在区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构。类的加载的最终产品是位于堆区中的Class对象,Class对象封装了类在方法区内的数据结构,并且向Java程序员提供了访问方法区内的数据结构的接口。

Java 中的所有类型包括基本类型(int, long, float等等),即使是数组都有与之关联的 Class 类的对象。

Class对象是由Jvm自动生成的,每当一个类被载入时,Jvm就自动为其生成一个Class对象

Class对象

实例.getClass()

通过Object的getClass()获取每一个实例对应的Class对象

String name = "hello";
Class stringClass = name.getClass();
System.out.println("类的名称:" + stringClass.getName());
System.out.println("是否为接口:" + stringClass.isInterface());
System.out.println("是否为基本类型:" + stringClass.isPrimitive());
System.out.println("是否为数组:" + stringClass.isArray());
System.out.println("父类名称:" + stringClass.getSuperclass().getName());
类的名称:java.lang.String
是否为接口:false是否为基本类型:false
是否为数组:false
父类名称:java.lang.Object

类名.class

你也可以直接使用一下方式来获取String类的Class对象

Class stringClass = String.class;

Class.forName()

在一些应用中,你无法事先知道使用者将载入什么类别,你可以使用Class的静态方法forName()来动态加载类别

Class c = Class.forName(args[0]);
System.out.println("类的名称:" + c.getName());
System.out.println("是否为接口:" + c.isInterface());
System.out.println("是否为基本类型:" + c.isPrimitive());
System.out.println("是否为数组:" + c.isArray());
System.out.println("父类名称:" + c.getSuperclass().getName());
$ java Demo1 java.util.Scanner  
类的名称:java.util.Scanner
是否为接口:false
是否为基本类型:false
是否为数组:false
父类名称:java.lang.Object

Class.forName()有两个版本,上面的版本只指定了全限定类名,而另一个版本可以让你指定类名,载入时是否执行静态代码块,执行类加载器(ClassLoader)

static Class forName(String name, boolean initialize, ClassLoader loader)
ClassLoader loader = Thread.currentThread().getContextClassLoader();
// Class.forName() 加载类 默认会执行初始化块
Class.forName("Test2");
// Class.forName() 加载类 第二个参数 可以控制是否执行初始化块
Class.forName("Test2", false, loader);

class Test2 {   
      static {
        System.out.println("静态初始化块执行了!");   
     }
 }

从Class对象中获取信息

Class对象表示所载入的类别,获取Class对象后,你就可以获取类别相关的信息,入 package, constructor, field, method等信息。
而每一种信息,都有相对应的类别

  • package: java.lang.reflect.Package
  • constructor: java.lang.reflect.Constructor
  • field: java.lang.reflect.Field
  • method: java.lang.reflect.Method
Class c = Class.forName(args[0]);
System.out.println("包信息package:" + c.getPackage());
System.out.println("类修饰符modifier:" + c.getModifiers());
System.out.println("构造方法constructor:");
Arrays.stream(c.getDeclaredConstructors()).forEach(System.out::println);
System.out.println("成员变量fields:");
Arrays.stream(c.getDeclaredFields()).forEach(System.out::println);
$ java Demo1 java.util.ArrayList
包信息package:package java.util
类修饰符modifier:1
构造方法constructor:
public java.util.ArrayList(java.util.Collection)
public java.util.ArrayList()
public java.util.ArrayList(int)
成员变量fields:
private static final long java.util.ArrayList.serialVersionUID
private static final int java.util.ArrayList.DEFAULT_CAPACITY
private static final java.lang.Object[] java.util.ArrayList.EMPTY_ELEMENTDATA
private static final java.lang.Object[] java.util.ArrayList.DEFAULTCAPACITY_EMPTY_ELEMENTDATA
transient java.lang.Object[] java.util.ArrayList.elementDataprivate
private int java.util.ArrayList.sizeprivate
private static final int java.util.ArrayList.MAX_ARRAY_SIZE

ClassLoader 类加载器

Java在需要使用类的时候才会将类载入,Java中类的载入是由Class Loader来实现的.

当你尝试执行java xxx命令时,java会尝试找到JRE的安装目录,然后寻找
jvm.dll,接着启动JVM并进行初始化操作,接着产生
BootstrapLoader,Bootstrap Loader会载入Extended Loader, 并设定Extended Loader 的parent 为 BootstrapLoader, 接着Bootstrap Loader 会载入 Application Loader, 并将Application Loader 的parent 设定为
Extended Loader

启动类加载器

BootstrapLoader搜寻 sun.boot.library.path中指定的类, 你可以使用
System.getProperty("sun.boot.library.path")来获取

/Library/Java/JavaVirtualMachines/jdk1.8.0_151.jdk/Contents/Home/jre/lib

扩展类加载器

Extended Loader(sun.misc.Launcher$ExtClassLoader) 是由Java编写的,会搜寻系统参数java.ext.dirs中指定的类别,可以通过System.getProperty("java.ext.dirs")来获取

/Users/dsying/Library/Java/Extensions:
/Library/Java/JavaVirtualMachines/jdk1.8.0_151.jdk/Contents/Home/jre/lib/ext:
/Library/Java/Extensions
:/Network/Library/Java/Extensions:
/System/Library/Java/Extensions:/usr/lib/java

应用程序类加载器

Application Loader
(sun.misc.Launcher$AppClassLoader) 是由Java编写的,会搜寻系统参数java.class.path中指定的类别,可以通过System.getProperty("java.class.path")来获取, 在使用java xxx命令执行.class字节码文件时,可以通过-cp参数设定classpath

java –cp ./classes SomeClass

类加载器之间的关系

ClassLoader loader Thread.currentThread().getContextClassLoader();
// sun.misc.Launcher$AppClassLoader@18b4aac2   应用类加载器
System.out.println(loader);
// sun.misc.Launcher$ExtClassLoader@610455d6   扩展类加载器
System.out.println(loader.getParent());
// Bootstrap ClassLoader 启动类加载器(用C语言实现,所以此处返回null)
System.out.println(loader.getParent().getParent());
sun.misc.Launcher$AppClassLoader@18b4aac2sun.misc.Launcher$ExtClassLoader@610455d6null复制代码

类加载有三种方式:

  1. 命令行启动应用时候由JVM初始化加载
  2. 通过Class.forName()方法动态加载
  3. 通过ClassLoader.loadClass()方法动态加载
Class.forName()和ClassLoader.loadClass()区别
  • Class.forName():将类的.class文件加载到jvm中之外,还会对类进行解释,执行类中的static块;
  • ClassLoader.loadClass():只干一件事情,就是将.class文件加载到jvm中,不会执行static中的内容,只有在newInstance才会去执行static块。
  • Class.forName(name, initialize, loader)带参函数也可控制是否加载static块。并且只有调用了newInstance()方法采用调用构造函数,创建类的对象。

JVM类加载机制

  • 全盘负责,当一个类加载器负责加载某个Class时,该Class所依赖的和引用的其他Class也将由该类加载器负责载入,除非显示使用另外一个类加载器来载入
  • 父类委托,先让父类加载器试图加载该类,只有在父类加载器无法加载该类时才尝试从自己的类路径中加载该类
  • 缓存机制,缓存机制将会保证所有加载过的Class都会被缓存,当程序中需要使用某个Class时,类加载器先从缓存区寻找该Class,只有缓存区不存在,系统才会读取该类对应的二进制数据,并将其转换成Class对象,存入缓存区。这就是为什么修改了Class后,必须重启JVM,程序的修改才会生效

双亲委派模型

双亲委派模型的工作流程是:如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把请求委托给父加载器去完成,依次向上,因此,所有的类加载请求最终都应该被传递到顶层的启动类加载器中,只有当父加载器在它的搜索范围中没有找到所需的类时,即无法完成该加载,子加载器才会尝试自己去加载该类。

双亲委派机制:

  1. AppClassLoader加载一个class时,它首先不会自己去尝试加载这个类,而是把类加载请求委派给父类加载器ExtClassLoader去完成。
  2. ExtClassLoader加载一个class时,它首先也不会自己去尝试加载这个类,而是把类加载请求委派给BootStrapClassLoader去完成。
  3. 如果BootStrapClassLoader加载失败(例如在$JAVA_HOME/jre/lib里未查找到该class),会使用ExtClassLoader来尝试加载;
  4. ExtClassLoader也加载失败,则会使用AppClassLoader来加载,如果AppClassLoader也加载失败,则会报出异常ClassNotFoundException

ClassLoader源码分析:

public Class<?> loadClass(String name)throws ClassNotFoundException { 
   return loadClass(name, false);
}
protected synchronized Class<?> loadClass(String name, boolean resolve)throws ClassNotFoundException {   
 // 首先判断该类型是否已经被加载   
 Class c = findLoadedClass(name); 
   if (c == null) {       
 //如果没有被加载,就委托给父类加载或者委派给启动类加载器加载     
   try {       
     if (parent != null) {            
    //如果存在父类加载器,就委派给父类加载器加载            
    c = parent.loadClass(name, false);         
   } else {           
     //如果不存在父类加载器,就检查是否是由启动类加载器加载的类,通过调用本地方法native Class findBootstrapClass(String name)          
      c = findBootstrapClass0(name);            }    
    } catch (ClassNotFoundException e) {          
      //如果父类加载器和启动类加载器都不能完成加载任务,才调用自身的加载功能            
    c = findClass(name);        
          }    
      }  
  if (resolve) {       
 resolveClass(c);  
  }   
 return c;}

双亲委派模型意义:

  • 系统类防止内存中出现多份同样的字节码
  • 保证Java程序安全稳定运行

自定义加载器

自定义类加载器一般都是继承自ClassLoader类,从上面对loadClass方法来分析来看,我们只需要重写 findClass 方法即可。下面我们通过一个示例来演示自定义类加载器的流程:

package com.github.hcsp.classloader;
import java.io.ByteArrayOutputStream;
import java.io.File;import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;

public class MyClassLoader extends ClassLoader {    
// 存放字节码文件的目录    
private final File bytecodeFileDirectory;   

 public MyClassLoader(File bytecodeFileDirectory) {   
     this.bytecodeFileDirectory = bytecodeFileDirectory;  
  }   
 // 还记得类加载器是做什么的么? 
   // "从外部系统中,加载一个类的定义(即Class对象)"  
  // 请实现一个自定义的类加载器,将当前目录中的字节码文件加载成为Class对象  
  // 提示,一般来说,要实现自定义的类加载器,你需要覆盖以下方法,完成:  
  //  
  // 1.如果类名对应的字节码文件存在,则将它读取成为字节数组   
 //   1.1 调用ClassLoader.defineClass()方法将字节数组转化为Class对象  
  // 2.如果类名对应的字节码文件不存在,则抛ClassNotFoundException  
  //   
 @Override  
  protected Class<?> findClass(String name) throws ClassNotFoundException {       
 byte[] classData = getByteArrayFromFile(name);      
  if (classData == null) {       
     throw new ClassNotFoundException();     
   }     
   return defineClass(name, classData, 0, classData.length); 
   }   
 byte[] getByteArrayFromFile(String className) throws ClassNotFoundException {      
  ByteArrayOutputStream bos = new ByteArrayOutputStream();      
  File file = new File(bytecodeFileDirectory, className + ".class");   
    int len = 0;    
    try { 
           byte[] bufferSize = new byte[1024];         
         FileInputStream fis = new FileInputStream(file);        
          while ((len = fis.read(bufferSize)) != -1) { 
               bos.write(bufferSize, 0, len);   
         }       
 } catch (FileNotFoundException e) {
            throw new ClassNotFoundException();  
      } catch (IOException e) {
            e.printStackTrace(); 
       } 
       return bos.toByteArray();
    } 

   public static void main(String[] args) throws Exception {
        File projectRoot = new File(System.getProperty("basedir", System.getProperty("user.dir")));
        MyClassLoader myClassLoader = new MyClassLoader(projectRoot);
        Class testClass = myClassLoader.loadClass("com.github.hcsp.MyTestClass");
        Object testClassInstance = testClass.getConstructor().newInstance();
        String message = (String) testClass.getMethod("sayHello").invoke(testClassInstance);
        System.out.println(message);    }}

自定义类加载器的核心在于对字节码文件的获取,如果是加密的字节码则需要在该类中对文件进行解密。由于这里只是演示,我并未对class文件进行加密,因此没有解密的过程.

其它

使用反射创建对象

你可以使用Class的newInstance()方法来实例化

Class c = Class.forName(className);
Object obj = c.newInstance();复制代码

调用方法

使用反射可以取回类中的方法,方法对应的类为
java.lang.reflect.Method, 你可以使用它的
invoke()方法来调用指定的方法

Class testClass = myClassLoader.loadClass("com.github.hcsp.MyTestClass");
Object testClassInstance = testClass.getConstructor().newInstance();
String message = (String) testClass.getMethod("sayHello").invoke(testClassInstance);

更多大厂面试资料以及Android资料可以进群一起交流:(Android进阶学习⑥群:345659112)
作者:sheng_ding
链接:https://juejin.cn/post/6844904154096205837
来源:稀土掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,377评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,390评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,967评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,344评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,441评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,492评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,497评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,274评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,732评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,008评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,184评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,837评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,520评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,156评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,407评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,056评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,074评论 2 352

推荐阅读更多精彩内容