Q:Java一次编译处处运行如何实现的
Java源码首先被编译成字节码(可以用javap指令查看字节码文件),再由不同平台的JVM进行解析,Java语言在不同的平台上运行时不需要进行重新编译,Java虚拟机在执行字节码的时候,把字节码转换成具体平台上的机器指令。
为什么这么做呢?
如果直接编译成字节码,每次执行都需要检查各种语法语义。
同时保证了兼容性,将别的语言解析成字节码。
Q:JVM如何加载class文件,JVM组成
JAM通过Class Loader将符合格式要求的class字节码文件加载到内存中,字节码文件中不同的类别的数据存储在内存不同的位置上。然后通过解释器对命令进行解析,并提交给操作系统去执行。
Class Loader:依据特定格式,加载class文件到内存
Execution Engine:对命令进行解析
Native Interface:融合不同开发语言的原生库为Java所用
Runtime Data Area: JVM内存空间结构模型
Q:Java反射
在运行状态中,对于任意一个类,都能够获取到这个类的所有属性和方法,对于任意一个对象,都能够调用它的任意一个方法和属性(包括私有的方法和属性),这种动态获取的信息以及动态调用对象的方法的功能就称为java语言的反射机制。通俗点讲,通过反射,该类对我们来说是完全透明的,想要获取任何东西都可以。
想要使用反射机制,就必须要先获取到该类的字节码文件对象(.class),通过字节码文件对象,就能够通过该类中的方法获取到我们想要的所有信息(方法,属性,类名,父类名,实现的所有接口等等),每一个类对应着一个字节码文件也就对应着一个Class类型的对象,也就是字节码文件对象。
1. 获取字节码对象有3种方式:
//第一种,forName()
Class clazz = Class.forName("swap.Test");
//第二种,class属性
Class clazz2 = Test.class;
//第三种,getClass()
Test t = new Test();
Class clazz3 = t.getClass();
2. 通过反射创建对象:
//第一种,newInstance()
Test test = (Test)clazz.newInstance();
//第二种,通过默认或者指定的构造方法获取对象
Constructor<Test> constructor = clazz.getDeclaredConstructor();
Test test2 = constructor.newInstance();
3. 通过反射获取方法:
//通过clazz的getDeclaredMethod获取到公有和私有方法,但是不能获取继承或实现接口的方法
Method m1 = clazz.getDeclaredMethod("privateMethod",String.class);
//私有方法设置true
m1.setAccessible(true);
m1.invoke(test,"Bob");
//可以获取共有方法,继承或实现接口的方法,但是不能获取私有方法
Method m2 = clazz.getMethod("publicMethod", String.class);
m2.invoke(test,"Mike");
4. 通过反射获取属性:
Field name = clazz.getDeclaredField("name");
name.setAccessible(true);
name.set(test,"Alice");
//或者声明Field[]获取多个属性