1. 类加载器
类加载器作用:负责将.class文件(存储的物理文件)加载到内存中,供虚拟机运行
类加载的时机:当需要用到某个类的时候就加载
1.1 类加载的过程
加载:通过包名+类名,在硬盘中找到对应字节码文件,以流的形式加载进内存。java一切都是对象,将字节码文件以对象的形式存开辟空间进行存储。
验证:确保Class文件字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身安全
准备:初始化静态变量
简析:本类中如果用到了其他类,此时就需要在内存中找到对应的类
初始化:根据程序员通过程序制定的主观计划去初始化类变量和其他资源
2.2 类加载器的分类
- Bootstrap class loader:虚拟机的内置类加载器,通常表示为null ,并且没有父null
- Platform class loader:平台类加载器,负责加载JDK中一些特殊的模块
- System class loader:系统类加载器,负责加载用户类路径上所指定的类库
双亲委派模式:
如果一个类加载器收到了类加载请求,它并不会自己先去加载,而是把这个请求委托给父类的加载器去执行,如果父类加载器还存在其父类加载器,则进一步向上委托,依次递归,请求最终将到达顶层的启动类加载器,如果父类加载器可以完成类加载任务,就成功返回,倘若父类加载器无法完成此加载任务,子加载器才会尝试自己去加载,这就是双亲委派模式
public class ClassLoaderDemo01 {
public static void main(String[] args) {
//获取系统类加载器
ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();
//获取系统类加载器的父加载器----平台类加载器
ClassLoader loaderParent = systemClassLoader.getParent();
//获取平台类加载器的父加载器-----启动类加载器
ClassLoader loaderParent1 = loaderParent.getParent();
System.out.println("系统:"+systemClassLoader);
System.out.println("平台:"+loaderParent);
System.out.println("启动:"+loaderParent1);//一般用null表示
}
}
2.3 加载器常用方法
方法名 | 说明 |
---|---|
public static ClassLoader getSystemClassLoader() | 获取系统类加载器 |
public InputStream getResourceAsStream(String name) | 加载某一个资源文件 |
import java.io.IOException;
import java.io.InputStream;
import java.util.Properties;
public class ClassLoaderDemo02 {
public static void main(String[] args) throws IOException {
//获取一个类加载器
ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();
//利用加载器去加载一个指定的文件
//参数为文件路径(类加载器对应的相对路径在src文件下
InputStream is = systemClassLoader.getResourceAsStream("prop.properties");
//prop相当于一个map集合
Properties prop = new Properties();
//load方法---从输入字节流中读取属性列表(键和值)
prop.load(is);
System.out.println(prop);//{name=zhangsan, age=16}
}
}
3. 反射
Java反射机制:
- 利用反射可以无视修饰符获取类里面的所有属性和方法
- 利用反射可以先获取配置文件中的信息,动态获取信息并创建对象和调用方法
3.1 获取class对象
利用反射机制无视修饰符去调用类中的方法,需要获得此类的class对象。
三种方式获取class对象:
public class ReflectDemo01 {
public static void main(String[] args) throws ClassNotFoundException {
//利用class类中的静态方法forName("全类名")
Class clazz = Class.forName("com.yulu.feng.lei_dui_xiang.Student");
System.out.println(clazz);
//通过class属性获取
Class clazz1 = Student.class;
System.out.println(clazz1);
//利用对象的getClass()方法
Student student = new Student();
Class clazz2 = student.getClass();
System.out.println(clazz2);
}
}
3.2 反射获取构造方法并使用
直接上代码:
public class Student {
private String name;
private int age;
public String school;
private Student(String school){
this.school = school;
System.out.println("我是私有构造方法!");
}
public Student(String name, int age) {
this.name = name;
this.age = age;
System.out.println("我是有参构造!");
}
public Student() {
System.out.println("我是空参构造!");
}
}
获取构造方法:
import java.lang.reflect.Constructor;
public class ReflectDemo02 {
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException {
Class clazz = Class.forName("com.yulu.feng.lei_dui_xiang.Student");
//method01(clazz);
//method02(clazz);
//method03(clazz);
//method04(clazz);
}
private static void method04(Class clazz) throws NoSuchMethodException {
//获取指定的私有构造方法
Constructor constructor = clazz.getDeclaredConstructor(String.class);
System.out.println(constructor);
}
private static void method03(Class clazz) throws NoSuchMethodException {
//获取指定的公共构造方法
Constructor constructor = clazz.getConstructor();
System.out.println(constructor);
Constructor constructor1 = clazz.getConstructor(String.class, int.class);
System.out.println(constructor1);
}
private static void method02(Class clazz) {
//获取所有构造方法包括私有
Constructor[] constructors = clazz.getDeclaredConstructors();
for (Constructor constructor : constructors) {
System.out.println(constructor);
}
}
private static void method01(Class clazz) {
//获取所有的公共构造方法
Constructor[] constructors = clazz.getConstructors();
for (Constructor constructor : constructors) {
System.out.println(constructor);
}
}
}
使用构造方法创建对象:
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
public class ReflectDemo03 {
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
Class clazz = Class.forName("com.yulu.feng.lei_dui_xiang.Student");
//空参构造
Constructor constructor = clazz.getConstructor();
Student student = (Student) constructor.newInstance();
System.out.println(student);
//有参构造
Constructor constructor1 = clazz.getConstructor(String.class, int.class);
Student student1 = (Student) constructor1.newInstance("zhangsan", 16);
System.out.println(student1);
//私有构造
Constructor constructor2 = clazz.getDeclaredConstructor(String.class);
//被private修饰的成员不能直接使用
//如果通过反射强行获取并使用,需要临时取消访问检查
constructor2.setAccessible(true);
Student student2 = (Student) constructor2.newInstance("四川大学");
System.out.println(student2);
}
}
3.3 反射获取成员变量并使用
方法名 | 说明 |
---|---|
Field[] getFields() | 返回所有公共成员变量对象的数组 |
Field[] getDeclaredFields() | 返回所有成员变量对象的数组 |
Field getField(String name) | 返回单个公共成员变量对象 |
Field getDeclaredField(String name) | 返回单个成员变量对象 |
方法名 | 说明 |
---|---|
void set(Object obj, Object value) | 赋值 |
Object get(Object obj) | 获取值 |
public class Student {
public String name;
public int age;
public String gender;
private int money = 300;
@Override
public String toString() {
return "Student{" +
"name='" + name + '\'' +
", age=" + age +
", gender='" + gender + '\'' +
", money=" + money +
'}';
}
}
public class ReflectDemo2 {
public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, IllegalAccessException, InstantiationException {
// Object get(Object obj) 返回由该 Field表示的字段在指定对象上的值。
//method1();
//method2();
}
private static void method2() throws ClassNotFoundException, NoSuchFieldException, InstantiationException, IllegalAccessException {
//1.获取class对象
Class clazz = Class.forName("com.itheima.myreflect4.Student");
//2.获取成员变量Field的对象
Field field = clazz.getDeclaredField("money");
//3.取消一下访问检查
field.setAccessible(true);
//4.调用get方法来获取值
//4.1创建一个对象
Student student = (Student) clazz.newInstance();
//4.2获取指定对象的money的值
Object o = field.get(student);
//5.打印一下
System.out.println(o);
}
private static void method1() throws ClassNotFoundException, NoSuchFieldException, InstantiationException, IllegalAccessException {
// void set(Object obj, Object value):给obj对象的成员变量赋值为value
//1.获取class对象
Class clazz = Class.forName("com.itheima.myreflect4.Student");
//2.获取name这个Field对象
Field field = clazz.getField("name");
//3.利用set方法进行赋值.
//3.1先创建一个Student对象
Student student = (Student) clazz.newInstance();
//3.2有了对象才可以给指定对象进行赋值
field.set(student,"zhangsan");
System.out.println(student);
}
}
3.4 反射获取成员方法并使用
方法名 | 说明 |
---|---|
Method[] getMethods() | 返回所有公共成员方法对象的数组,包括继承的 |
Method[] getDeclaredMethods() | 返回所有成员方法对象的数组,不包括继承的 |
Method getMethod(String name, Class<?>... parameterTypes) | 返回单个公共成员方法对象 |
Method getDeclaredMethod(String name, Class<?>... parameterTypes) | 返回单个成员方法对象 |
public class Student {
//私有的,无参无返回值
private void show() {
System.out.println("私有的show方法,无参无返回值");
}
//公共的,无参无返回值
public void function1() {
System.out.println("function1方法,无参无返回值");
}
//公共的,有参无返回值
public void function2(String name) {
System.out.println("function2方法,有参无返回值,参数为" + name);
}
//公共的,无参有返回值
public String function3() {
System.out.println("function3方法,无参有返回值");
return "aaa";
}
//公共的,有参有返回值
public String function4(String name) {
System.out.println("function4方法,有参有返回值,参数为" + name);
return "aaa";
}
}
public class ReflectDemo1 {
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException {
//method1();
//method2();
//method3();
//method4();
//method5();
}
private static void method5() throws ClassNotFoundException, NoSuchMethodException {
//Method getDeclaredMethod(String name, Class<?>... parameterTypes):
// 返回单个成员方法对象
//1.获取class对象
Class clazz = Class.forName("com.itheima.myreflect5.Student");
//2.获取一个成员方法show
Method method = clazz.getDeclaredMethod("show");
//3.打印一下
System.out.println(method);
}
private static void method4() throws ClassNotFoundException, NoSuchMethodException {
//1.获取class对象
Class clazz = Class.forName("com.itheima.myreflect5.Student");
//2.获取一个有形参的方法function2
Method method = clazz.getMethod("function2", String.class);
//3.打印一下
System.out.println(method);
}
private static void method3() throws ClassNotFoundException, NoSuchMethodException {
//Method getMethod(String name, Class<?>... parameterTypes) :
// 返回单个公共成员方法对象
//1.获取class对象
Class clazz = Class.forName("com.itheima.myreflect5.Student");
//2.获取成员方法function1
Method method1 = clazz.getMethod("function1");
//3.打印一下
System.out.println(method1);
}
private static void method2() throws ClassNotFoundException {
//Method[] getDeclaredMethods():
// 返回所有成员方法对象的数组,不包括继承的
//1.获取class对象
Class clazz = Class.forName("com.itheima.myreflect5.Student");
//2.获取Method对象
Method[] methods = clazz.getDeclaredMethods();
//3.遍历一下数组
for (Method method : methods) {
System.out.println(method);
}
}
private static void method1() throws ClassNotFoundException {
// Method[] getMethods():返回所有公共成员方法对象的数组,包括继承的
//1.获取class对象
Class clazz = Class.forName("com.itheima.myreflect5.Student");
//2.获取成员方法对象
Method[] methods = clazz.getMethods();
//3.遍历
for (Method method : methods) {
System.out.println(method);
}
}
}
方法名 | 说明 |
---|---|
Object invoke(Object obj, Object... args) | 运行方法 |
public class ReflectDemo2 {
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InstantiationException, InvocationTargetException {
// Object invoke(Object obj, Object... args):运行方法
// 参数一:用obj对象调用该方法
// 参数二:调用方法的传递的参数(如果没有就不写)
// 返回值:方法的返回值(如果没有就不写)
//1.获取class对象
Class clazz = Class.forName("com.itheima.myreflect5.Student");
//2.获取里面的Method对象 function4
Method method = clazz.getMethod("function4", String.class);
//3.运行function4方法就可以了
//3.1创建一个Student对象,当做方法的调用者
Student student = (Student) clazz.newInstance();
//3.2运行方法
Object result = method.invoke(student, "zhangsan");
//4.打印一下返回值
System.out.println(result);
}
}
如有错误欢迎留言评论,及时更正。2021年6月18日 羽露风