1.2 类加载器
-
类加载器的作用
-
负责将.class文件加载到内存中, 并为之生成对应的java.lang.Class对象
-
虽然我们不用过分关心类加载机制, 但是了解这个机制我们就能更好的理解程序的运行
-
JVM的类加载机制
-
全盘负责: 就是当一个类加载器负责加载某个Class时,该Class所依赖的和引用的其他Class也将由该类加载器负责载入,除非显示使用另外一个类加载器来载入
-
父类委托: 就是当一个类加载器负责加载某个Class时,先让父类加载器试图加载该Class,只有在父类加载器无法加载该类时才尝试从自己的类路径中加载该类
-
缓存机制: 保证所有加载过的Class都会被缓存,当程序需要使用某个Class对象时, 载器先从缓存区中搜索该Class,只有当缓存区中不存在该Class对象时,系统才会读取该类对应的二进制数据,并将其转换成Class对象,存储到缓存区
-
ClassLoader:是负责加载类的对象
-
包:
java.lang.ClassLoader
-
ClassLoader 中的两个方法
| 方法名 | 说明 |
| — | — |
| static ClassLoader getSystemClassLoader() | 返回用于委派的系统类加载器 |
| ClassLoader getParent() | 返回父加载器进行委派 |
-
Java运行时具有以下内置类加载器
-
Bootstrap class loader:它是虚拟机的内置类加载器,通常表示为null,并且没有父null
-
Platform class loader:平台类加载器可以看到所有平台类,平台类包括由平台类加载器或其祖先定义的Java SE平台API,其实现类和JDK特定的运行时类
-
System class loader:它也被称为应用程序类加载器,与平台类加载器不同。系统类加载器通常用于定义应用程序类路径,模块路径和JDK特定工具上的类
-
类加载器的继承关系:System的父加载器为Platform,而Platform的父加载器为Bootstrap
package test;
public class Demo {
public static void main(String[] args) {
//1,static ClassLoader getSystemClassLoader() 返回用于委派的系统类加载器
ClassLoader c = ClassLoader.getSystemClassLoader();
System.out.println©; //AppClassLoader
//2,ClassLoader getParent() 返回父加载器进行委派
ClassLoader c1 = c.getParent();
System.out.println(c1); //JDK8:ExtClassLoader
ClassLoader c2 = c1.getParent();
System.out.println(c2); //null
}
}
2.1 反射概述
- Java反射机制:是指在运行时去获取一个类的变量和方法信息。然后通过获取到的信息来创建对象,调用方法的一种机制。由于这种动态性,可以极大的增强程序的灵活性,程序不用在编译期就完成确定,在运行期仍然可以扩展
2.2 反射获取Class类的对象
-
我们要想通过反射去使用一个类,首先我们要获取到该类的字节码文件对象,也就是类型为Class类型的对象这里我们提供三种方式获取Cass类型的对象
-
1、使用类的class属性来获取该类对应的Class对象。举例:
Student.class
将会返回Student类对应的Class对象 -
2、调用对象的
getClass()
方法, 返回该对象所属类对应的Class对象
该方法是Object类中的方法,所有的Java对象都可以调用该方法
- 3、使用Class类中的静态方法
forName(StringclassName)
, 该方法需要传入字符串参数,该字符串参数的值是某个类的全路径,也就是完整包名的路径
package test;
public class Demo {
public static void main(String[] args) throws ClassNotFoundException {
//1、使用类的class属性来获取该类对应的Class对象
Class c1 = Student.class;
System.out.println(c1); //class test.Student
Class c2 = Student.class;
System.out.println(c1 == c2); //true;一个类只有一个字节码文件对象
//2、调用对象的getClass()方法, 返回该对象所属类对应的Class对象
Student s = new Student();
Class<? extends Student> c3 = s.getClass();
System.out.println(c1 == c3); //true
//3、使用Class类中的静态方法forName(StringclassName)
Class<?> c4 = Class.forName(“test.Student”);
System.out.println(c1 == c4); //true
}
}
2.3 反射获取构造方法并使用
- Class类中用于获取构造方法的方法
| 方法名 | 说明 |
| — | — |
| Constructor<?>[] getConstructors()
| 返回所有公共构造方法对象的数组 |
| Constructor<?> [] getDeclaredConstructors
[dɪˈkleəd] | 返回所有构造方法对象的数组 |
| Constructor<T> getConstructor(Class<?>...parameterTypes)
| 返回单个公共构造方法对象 |
| Constructor<T> getDeclaredConstructor(Class<?> ...parameterTypes)
| 返回单个构造方法对象 |
- Constructor类中用来创建对象的方法
| 方法名 | 说明 |
| — | — |
| T newInstance(Object…initargs) | 根据指定的构造方法创建对象 |
package test;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
public class Demo {
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
//获取Class对象
Class<?> c = Class.forName(“test.Student”);
//1,Constructor<?>[] getConstructors() 返回所有公共构造方法对象的数组
Constructor<?>[] cons1 = c.getConstructors();
for(Constructor con:cons1) {
System.out.println(con);
// public test.Student(java.lang.String,int,java.lang.String)
// public test.Student()
}
System.out.println(“--------”);
//2,Constructor<?> [] getDeclaredConstructors 返回所有构造方法对象的数组
Constructor<?>[] cons2 = c.getDeclaredConstructors();
for(Constructor con:cons2) {
System.out.println(con);
// public test.Student(java.lang.String,int,java.lang.String)
// test.Student(java.lang.String,int)
// private test.Student(java.lang.String)
// public test.Student()
}
System.out.println(“--------”);
/*
一、首先把这个类的带包的字符串路径传给forName()方法得到字节码文件Class对象c
二、c调用getConstructors()得到单个的构造方法对象con1
三、通过con1调用newInstance()创建一个对象,这就叫反射*/
//3,Constructor getConstructor(Class<?>…parameterTypes) 返回单个公共构造方法对象
//参数是:你要获得的构造方法的参数个数和数据类型对应字节码文件
Constructor<?> con1 = c.getConstructor();
//Constructor提供了一个类的单个构造方法信息和访问权限
//5,T newInstance(Object…initargs) 根据指定的构造方法创建对象
Object obj1 = con1.newInstance();
System.out.println(obj1); //Student{name=‘null’, age=0, address=‘null’}
}
}
- 案例1:
package test;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
public class Demo {
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
//一、首先把这个类的带包的字符串路径传给Class类的forName()方法得到字节码文件Class对象c
Class<?> c = Class.forName(“test.Student”);
//public Student(String name, int age, String address)
// 二、c调用getConstructor()得到单个的构造方法对象con1
Constructor<?> con = c.getConstructor(String.class, int.class, String.class);//基本数据类型int也可以通过.class得到对应class类型
// 三、通过con调用newInstance()创建一个对象,这就叫反射
Object obj = con.newInstance(“小白”,10, “成都”);
System.out.println(obj); //Student{name=‘小白’, age=10, address=‘成都’}
}
}
- 案例2:
package test;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
public class Demo {
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
//一、首先把这个类的带包的字符串路径传给forName()方法得到字节码文件Class对象c
Class<?> c = Class.forName(“test.Student”);
//private Student(String name)
//二、通过c调用getConstructor()创建无参构造方法对象
Constructor<?> con = c.getDeclaredConstructor(String.class);
//三、通过con调用newInstance()创建一个对象
// Object obj = con.newInstance(“小黑”);
// System.out.println(obj); //IllegalAccessException:没法访问异常:private修饰的私有构造方法
//暴力反射:public void setAccessible(boolean flag):值为true,取消访问检查
con.setAccessible(true);
Object obj = con.newInstance(“小黑”);
System.out.println(obj); //Student{name=‘小黑’, age=0, address=‘null’}
}
}
2.4 反射获取成员变量并使用
- Class类中用于获取成员变量的方法
| 方法名 | 说明 |
| — | — |
| Field[ ] getFields() | 返回所有公共成员变量对象的数组 |
| Field[ ] getDeclaredFields() | 返回所有成员变量对象的数组 |
| Field getField(String name) | 返回单个公共成员变量对象 |
| Field getDeclaredField(String name) | 返回单个成员变量对象 |
- Field类中用于给成员变量赋值的方法
| 方法名 | 说明 |
| — | — |
| void set(Object obj, Object value) | 给obj对象的成员变量赋值为value |
package test;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
public class Demo {
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException, NoSuchFieldException {
//获取Class对象
Class<?> c = Class.forName(“test.Student”);
//1,Field[] getFields() 返回所有公共成员变量对象的数组
Field[] fields1 = c.getFields();
for(Field field:fields1) {
System.out.println(field);
//public java.lang.String test.Student.address
}
System.out.println(“--------”);
//2,Field[ ] getDeclaredFields() 返回所有成员变量对象的数组
Field[] fields2 = c.getDeclaredFields();
for(Field field:fields2) {
System.out.println(field);
// private java.lang.String test.Student.name
// int test.Student.age
// public java.lang.String test.Student.address
}
System.out.println(“--------”);
/*
一、首先把这个类的带包的字符串路径传给forName()方法得到字节码文件Class对象c
二、通过c调用getField()得到成员变量对象addressField
三、addressField调用set()方法给obj对象的address赋值*/
//3,Field getField(String name) 返回单个公共成员变量对象
Field addressField = c.getField(“address”);
System.out.println(addressField);
//public java.lang.String test.Student.address
//获取无参构造方法创建对象
Constructor<?> con = c.getConstructor();
Object obj = con.newInstance();
//5,void set(
必看视频!获取2024年最新Java开发全套学习资料 备注Java
Object obj, Object value) 给obj对象的成员变量赋值为value
addressField.set(obj,“成都”); //给obj的成员变量addressField》address赋值为成都
System.out.println(obj); //Student{name=‘null’, age=0, address=‘成都’}
}
}
- 案例:
package test;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
public class Demo {
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException, NoSuchFieldException {
//一、首先把这个类的带包的字符串路径传给forName()方法得到字节码文件Class对象c
Class<?> c = Class.forName(“test.Student”);
//二、通过c调用getConstructor()创建无参构造方法对象
Constructor<?> con = c.getConstructor();
//三、通过con调用newInstance()创建一个对象
Object obj = con.newInstance();
//四、通过c调用getDeclaredField()得到成员变量对象
Field nameField = c.getDeclaredField(“name”);
nameField.setAccessible(true); //取消私有变量的访问检查
//五、成员变量对象调用set()方法给obj对象的成员变量赋值
nameField.set(obj,“小黑”);
Field ageField = c.getDeclaredField(“age”);
ageField.setAccessible(true);
ageField.set(obj,10);
Field addressField = c.getDeclaredField(“address”);
addressField.setAccessible(true);
addressField.set(obj,“成都”);
System.out.println(obj); //Student{name=‘小黑’, age=10, address=‘成都’}
}
}
2.5 反射获取成员方法并使用
- Class类中获取成员方法的方法
| 方法名 | 说明 |
| — | — |
| Method[ ] getMethods() | 返回所有公共成员方法对象的数组,包括继承的 |
| Method[ ] getDeclaredMethods() | 返回所有成员方法对象的数组,不包括继承的 |
| Method getMethod(String name, Class<?> … parameterTypes) | 返回单个公共成员方法对象 |
| Method getDeclaredMethod(String name, Class<?> … parameterTypes) | 返回单个成员方法对象 |
- Method类中用于调用成员方法的方法
| 方法名 | 说明 |
| — | — |
| Object invoke(Object obj, Objet… args) | 调用obj对象的成员方法,参数是args,返回值是Object类型 |
package test;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public class Demo {
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
//获取Class对象
Class<?> c = Class.forName(“test.Student”);
//1,Method[ ] getMethods() 返回所有公共成员方法对象的数组,包括继承的
Method[] methods1 = c.getMethods();
for(Method method:methods1) {
System.out.println(method);
// public java.lang.String test.Student.toString()
// public void test.Student.method1()
// public void test.Student.method3()
// public void test.Student.method2()
// public final void java.lang.Object.wait() throws java.lang.InterruptedException
// public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException
// …
}
System.out.println(“---------”);
//2,Method[ ] getDeclaredMethods() 返回所有成员方法对象的数组,不包括继承的
Method[] methods2 = c.getDeclaredMethods();
for(Method method:methods2) {
System.out.println(method);
// public java.lang.String test.Student.toString()
// private void test.Student.function()
// public void test.Student.method1()
// public void test.Student.method3()
// public void test.Student.method2()
}
//3,Method getMethod(String name, Class<?> … parameterTypes) 返回单个公共成员方法对象
Method m = c.getMethod(“method1”);
//获取无参构造方法并创建对象
Constructor<?> con = c.getConstructor();
Object obj = con.newInstance();
//5,Object invoke(Object obj, Objet… args) 调用obj对象的成员方法,参数是args,返回值是Object类型
//第一个Object:返回值类型;obj:对象;args:方法需要的参数
m.invoke(obj); //method1
}
}
- 案例:
- Student类
package test;
public class Student {
//成员变量:一个私有、一个默认、一个公共
private String name;
int age;
public String address;
//构造方法:一个私有、一个默认、两个公共
public Student() {
}
private Student(String name) {
this.name = name;
}
Student(String name, int age) {
this.name = name;
this.age = age;
}
public Student(String name, int age, String address) {
this.name = name;
this.age = age;
this.address = address;
}
//成员方法:一个私有,四个公共
private void function() {
System.out.println(“function”);
}
public void method1() {
System.out.println(“method1”);
}
public void method2(String s) {
System.out.println(“method2:” + s);
}
public String method3(String s, int i) { return s + “,” + i; }
@Override
public String toString() {
return “Student{” +
“name='” + name + ‘’’ +
“, age=” + age +
“, address='” + address + ‘’’ +
‘}’;
}
}
- 测试类
package test;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public class Demo {
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
//一、首先把这个类的带包的字符串路径传给forName()方法得到字节码文件Class对象c
Class<?> c = Class.forName(“test.Student”);
//二、通过c调用getConstructor()创建无参构造方法对象
Constructor<?> con = c.getConstructor();
//三、通过con调用newInstance()创建一个对象
Object obj = con.newInstance();
//四、通过字节码文件对象调用getMethod()方法得到method方法对象
Method m1 = c.getMethod(“method1”);
Method m2 = c.getMethod(“method2”, String.class);
Method m3 = c.getMethod(“method3”, String.class, int.class);
Method m4 = c.getDeclaredMethod(“function”);
//五、通过method方法对象调用invoke()方法
m1.invoke(obj); //method1
m2.invoke(obj,“小黑”); //method2:小黑
Object o = m3.invoke(obj, “小白”, 10);
System.out.println(o); //小白,10
m4.setAccessible(true);
总结
总的来说,面试是有套路的,一面基础,二面架构,三面个人。
最后,小编这里收集整理了一些资料,其中包括面试题(含答案)、书籍、视频等。希望也能帮助想进大厂的朋友
cessException, InvocationTargetException, InstantiationException {
//一、首先把这个类的带包的字符串路径传给forName()方法得到字节码文件Class对象c
Class<?> c = Class.forName(“test.Student”);
//二、通过c调用getConstructor()创建无参构造方法对象
Constructor<?> con = c.getConstructor();
//三、通过con调用newInstance()创建一个对象
Object obj = con.newInstance();
//四、通过字节码文件对象调用getMethod()方法得到method方法对象
Method m1 = c.getMethod(“method1”);
Method m2 = c.getMethod(“method2”, String.class);
Method m3 = c.getMethod(“method3”, String.class, int.class);
Method m4 = c.getDeclaredMethod(“function”);
//五、通过method方法对象调用invoke()方法
m1.invoke(obj); //method1
m2.invoke(obj,“小黑”); //method2:小黑
Object o = m3.invoke(obj, “小白”, 10);
System.out.println(o); //小白,10
m4.setAccessible(true);
总结
总的来说,面试是有套路的,一面基础,二面架构,三面个人。
最后,小编这里收集整理了一些资料,其中包括面试题(含答案)、书籍、视频等。希望也能帮助想进大厂的朋友
[外链图片转存中…(img-aZlhxvzP-1716462086714)]
[外链图片转存中…(img-9b1YfQVW-1716462086714)]