Java基础——类加载器、反射、动态代理

1.类的加载和加载时机

类的加载:
	当程序要使用某个类时,如果该类还未被加载到内存中,
	则系统会通过加载,连接,初始化三步来实现对这个类进行初始化。
	加载 
		就是指将class文件读入内存,并为之创建一个Class对象。 
		任何类被使用时系统都会建立一个Class对象。
	连接
		验证 :	是否有正确的内部结构,并和其他类协调一致
		准备 :	负责为类的静态成员分配内存,并设置默认初始化值
		解析:	把类中的符号引用转换为直接引用
	初始化	

类的加载时机:
创建类的实例 new Student()
访问类的静态变量,或者为静态变量赋值 Math.PI Math.class
调用类的静态方法 Math.abs()
使用反射方式来强制创建某个类或接口对应的java.lang.Class对象
初始化某个类的子类
直接使用java.exe命令来运行某个主类

2.类加载器

类加载器:负责将.class文件加载到内在中,并为之生成对应的Class对象。

分类:
	Bootstrap ClassLoader 根类加载器
	Extension ClassLoader 扩展类加载器
	Sysetm ClassLoader 系统类加载器

类加载器的作用:
	Bootstrap ClassLoader 根类加载器
		也被称为引导类加载器,负责Java核心类的加载  
		比如System,String等。在JDK中JRE的lib目录下rt.jar文件中  
	Extension ClassLoader 扩展类加载器
		负责JRE的扩展目录中jar包的加载。
		在JDK中JRE的lib目录下ext目录
	Sysetm ClassLoader 系统类加载器  
		负责在JVM启动时加载来自java命令的class文件,以及classpath环境变量所指定的jar包和类路径
双亲委派机制:
	工作原理:
	1)如果一个类加载器收到了类加载请求,它并不会自己先去加载,而是把这个请求委托给父类的
	加载器去执行;
	2)如果父类加载器还存在其父类加载器,则进一步向上委托,依次递归,请求最终将到达项层的启
	动类加载器;
	3)如果父类加载器可以完成类加载任务,就成功返回,倘若父类加载器无法完成此加载任务,子
	加载器才会尝试自己去加载,这就是双亲委派机制。

什么要采用双亲委派机制
因为如果随意的就去执行我自己写的String了,那么你说你的一个很完美的项目,我要是恶意攻击
你给你发送一个String类,那么你的项目岂不是直接就崩溃了?这也是一种保护啊!
1)避免核心API被篡改
2)避免类的重复加载

3.反射

概述:JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;
对于任意一个对象,都能够调用它的任意一个方法和属性;
这种动态获取类的信息以及动态调用对象的方法的功能称为java语言的反射机制。
要想解剖一个类,必须先要获取到该类的字节码文件对象。
而解剖使用的就是Class类中的方法,所以先要获取到每一个字节码文件对应的Class类型的对象

获取class文件对象的三种方式
Object类的getClass()方法
静态属性class
Class类中静态方法forName()

案例演示:
获取class文件对象的三种方式
		反射: 就是在运行状态中的一种动态调用方法或者属性的一种机制.
		- 就是获取字节码文件对象,然后剖析该类中存在哪些构造方法,哪些成员变量,哪些成员方法
- 类的成员
- 成员变量		Field
- 构造方法		Constructor
- 成员方法		Method
- 如何获取一个类对应的字节码文件对象:
- a: 第一种通过Object类中的getClass方法
- b: 通过静态属性(class属性)
- c: 通过Class类中的一个静态方法:
- public static Class forName(String className): 
- className: 这个表示的是一个类对应的全类名(就是需要加上包名)
public class MyTest {
    public static void main(String[] args) throws ClassNotFoundException {
        //我要使用反射机制,先要获取改类的字节码文件对象
        //获取一个类的字节码文件对象,有以下几种方式

        //方式1
        Student student = new Student();
        Class<? extends Student> aClass = student.getClass();

        //方式2:通过一个类全限定类名可以获取该类的字节码文件对象
        Class<?> aClass1 = Class.forName("org.westos.demo.Student");

        //方式3:每个类都有一个 class属性,通过该属性就可以获取该类的字节码文件对象
        Class<Student> studentClass = Student.class;
        System.out.println(aClass);
        System.out.println(aClass1);
        System.out.println(studentClass);
        System.out.println(aClass==aClass1);
        System.out.println(aClass1==studentClass);
    }
}

4.通过反射获取无参构造方法并使用

获取所有构造方法
public Constructor<?>[] getConstructors() 获取所有的公共的构造方法不包含私有的
public Constructor<?>[] getDeclaredConstructors() 获取所有的构造方法 包括私有的

获取单个构造方法
public Constructor getConstructor(Class<?>… parameterTypes) 获取单个的公共的构造方法 不包含私有的
public Constructor getDeclaredConstructor(Class<?>… parameterTypes) 获取单个的构造方法包含私有的

public class MyTest {
    public static void main(String[] args) throws ClassNotFoundException {
        //一个类的构成:构造方法,成员变量,成员方法
        //构造方法 Constructor
        //成员变量 Field
        //成员方法   Method
        //通过反射来调用类中的构造方法执行。

        //获取该类的字节码文件对象
        Class<?> aClass = Class.forName("com.xawl.demo2.Student");
        //获取该类的所有的公共的构造方法对象
        Constructor<?>[] constructors = aClass.getConstructors();
        for (Constructor<?> constructor : constructors) {
            System.out.println(constructor);
        }
        System.out.println("========================");
        //获取该类的所有的构造方法对象,包括私有的
        Constructor<?>[] declaredConstructors = aClass.getDeclaredConstructors();
        for (Constructor<?> declaredConstructor : declaredConstructors) {
            System.out.println(declaredConstructor);
        }
    }
}
public class MyTest2 {
    public static void main(String[] args) throws NoSuchMethodException, ClassNotFoundException {
        Class<?> aClass = Class.forName("com.xawl.demo2.Student");
        //获取单个的构造方法对象
        Constructor<?> constructor = aClass.getConstructor();//获取空参的构造方法对象
        System.out.println(constructor);

        //获取一个参数的构造方法对象 参数传入,你构造方法的参数的字节码类型
        Constructor<?> constructor1 = aClass.getConstructor(String.class);
        System.out.println(constructor1);

        Constructor<?> constructor2 = aClass.getConstructor(String.class,int.class);
        System.out.println(constructor2);

        //获取私有的构造方法对象
        Constructor<?> constructor3 = aClass.getDeclaredConstructor(String.class, int.class,char.class);
        System.out.println(constructor3);

    }
}

5.通过反射获取私有构造方法并使用

通过反射获取私有构造方法并使用
// 获取字节码文件对象
Class clazz = Class.forName(“com.click369.Student”) ;
Constructor con = clazz.getDeclaredConstructor(String.class , int.class) ;
// 值为 true 则指示反射的对象在使用时应该取消 Java 语言访问检查。
con.setAccessible(true) ; //取消语法检查不然会报错 因为私有的外界不能直接访问
Object obj = con.newInstance(“张三” , 23) ;
System.out.println(obj);

public class MyTest3 {
    public static void main(String[] args) throws IllegalAccessException, InvocationTargetException, InstantiationException, NoSuchMethodException,
            ClassNotFoundException {
        //以前的方式,要创建一个类的对象,得调用构造方法
        //Student student = new Student();

        //现在我们通过反射的方式,来创建一个类的对象
        Class<?> aClass = Class.forName("com.xawl.demo2.Student");
        //获取空参的构造方法对象
        Constructor<?> constructor = aClass.getConstructor();

        Object obj = constructor.newInstance();
        Student student= (Student) obj;
        System.out.println(student);
        System.out.println("======================================");
        //通过有参构造来创建对象
        //以前的方式
       // Student student1 = new Student("张三");
        //获取一个参数的构造方法对象
        Constructor<?> constructor1 = aClass.getConstructor(String.class);
        Object student2 = constructor1.newInstance("张三");

        System.out.println(student2);


        System.out.println("===========================================");
        //通过两个参数的构造方法创建对象
        //以前的方式
        //Student student1 = new Student("王五", 25);
        // 获取两个参数的构造方法对象
        Constructor<?> constructor2 = aClass.getConstructor(String.class, int.class);

        Object s3 = constructor2.newInstance("王五", 25);

        System.out.println(s3);

        System.out.println("=======================================");
        //通过私有的构造方法创建对象,以前的方式是不可以的
       // Student student1 = new Student("赵六", 26, '男');

        //通过反射的方式就可以

        Constructor<?> declaredConstructor = aClass.getDeclaredConstructor(String.class, int.class, char.class);
        //取消私有权限的检查
        declaredConstructor.setAccessible(true);
        Student s4 = (Student) declaredConstructor.newInstance("赵六", 26, '男');
        System.out.println(s4);
    }
}

6.通过反射获取成员变量并使用

获取所有成员变量
public Field[] getFields() 获取所有公共的成员变量包含从父类继承过来的
public Field[] getDeclaredFields() 获取所有的成员变量 包含私有的 也包含从父类继承过来的成员变量

获取单个成员变量
public Field getField(String name)
public Field getDeclaredField(String name)

获取所有成员变量.java

public class MyTest {
    public static void main(String[] args) throws ClassNotFoundException {
        //使用反射机制,操作成员变量
       // Field 描述的是成员变量类型
        Class<?> aClass = Class.forName("com.xawl.demo3.Teacher");
        //获取所有的公共的成员变量对象
        Field[] fields = aClass.getFields();
        for (Field field : fields) {
            System.out.println(field);
        }

        System.out.println("=================================");

        //获取所有的公共的成员变量对象,包括私有的成员变量
        Field[] declaredFields = aClass.getDeclaredFields();
        for (Field declaredField : declaredFields) {
            System.out.println(declaredField);
        }
    }
}

获取单个成员变量.java

public class MyTest2 {
    public static void main(String[] args) throws NoSuchFieldException {
        //获取单个的公共的成员变量对象
        Class<Teacher> teacherClass = Teacher.class;
        Field name = teacherClass.getField("name");
        System.out.println(name);

        //获取私有的成员变量对象
        Field age = teacherClass.getDeclaredField("age");
        System.out.println(age);

    }
}

案例演示:通过反射获取成员变量并使用

public class MyTest3 {
    public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException, InstantiationException {
        //以前的方式
        //Teacher teacher = new Teacher();
        //teacher.name="abc";

        //反射的方式
        Class<Teacher> teacherClass = Teacher.class;
        //Teacher teacher1 = teacherClass.getConstructor().newInstance();
        Teacher teacher = teacherClass.newInstance();
        Field nameObj = teacherClass.getField("name");
        nameObj.set(teacher,"张三");
        String name = (String) nameObj.get(teacher);
        System.out.println(name);
        System.out.println("==========================");
        Field salObj = teacherClass.getDeclaredField("sal");
        salObj.setDouble(teacher,3.52);
        double aDouble = salObj.getDouble(teacher);
        System.out.println(aDouble);
        System.out.println("=======================================");
        //获取私有的字段对象
        Field ageObj = teacherClass.getDeclaredField("age");
        ageObj.setAccessible(true); //取消私有权限的检查
        ageObj.setInt(teacher,100);
        int anInt = ageObj.getInt(teacher);
        System.out.println(anInt);
    }
}

7.通过反射获取无参无返回值成员方法并使用

获取所有成员方法
public Method[] getMethods() //获取所有的公共的成员方法不包含私有的 包含从父类继承过来的过来的公共方法
public Method[] getDeclaredMethods()//获取自己的所有成员方法 包含私有的

获取单个成员方法
//参数1: 方法名称 参数2:方法行参的class 对象
public Method getMethod(String name,Class<?>… parameterTypes) //获取单个的方法 不包含私有的
public Method getDeclaredMethod(String name,Class<?>… parameterTypes) 获取单个方法包括私有的

public class MyTest {
    public static void main(String[] args) throws ClassNotFoundException {
        Class<?> aClass = Class.forName("com.xawl.demo.Student");
        //获取所有的公共的方法对象
        //Method 用来描述方法对象的,包括父类的
        Method[] methods = aClass.getMethods();
        for (Method method : methods) {
            System.out.println(method);
        }
        System.out.println("========================");

        //获取所有的方法对象,包括私有的,但是不获取父类的
        Method[] declaredMethods = aClass.getDeclaredMethods();
        for (Method declaredMethod : declaredMethods) {
            System.out.println(declaredMethod);
        }
    }
}
public class MyTest2 {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException {
        Class<?> aClass = Class.forName("org.westos.demo.Student");
        //获取公共的方法对象,参数就方法名
        Method show = aClass.getMethod("show");
        System.out.println(show);

        Method test = aClass.getMethod("test", String.class);
        System.out.println(test);


        Method add = aClass.getMethod("add",int.class,int.class);
        System.out.println(add);

        //获取私有的方法对象

        Method method = aClass.getDeclaredMethod("method", int.class);
        System.out.println(method);

    }
}

通过反射获取带参带返回值成员方法并使用

public class MyTest3 {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, IllegalAccessException,
            InstantiationException {
        //以前我们调用方法的方式 ,new对象调用方法
        //Student student = new Student();
        //student.show();

        //反射的方式 ,调用方法执行
        Class<?> aClass = Class.forName("com.xawl.demo.Student");
        Object obj = aClass.newInstance();
        //获取公共的方法对象,参数就方法名
        Method show = aClass.getMethod("show");
        show.invoke(obj);

        System.out.println("=============================");
        Method test = aClass.getMethod("test", String.class);
        test.invoke(obj,"张三");
        System.out.println("============================");
        Method add = aClass.getMethod("add", int.class, int.class);
        //返回值就是,你调用完方法返回的值
        Integer invoke = (Integer) add.invoke(obj, 20, 30);
        System.out.println(invoke);

        Method method = aClass.getDeclaredMethod("method", int.class);
        method.setAccessible(true);//取消私有的权限检查
        Object invoke1 = method.invoke(obj, 20);
        System.out.println(invoke1);
    }
}

8.通过反射运行配置文件内容

properties配置文件
在这里插入图片描述

public class MyTest {
    public static void main(String[] args) throws Exception {
        //Dog dog = new Dog();
        //dog.eat();
        //Cat cat = new Cat();
        //cat.eat();
        Properties properties = new Properties();
        properties.load(new FileInputStream("src/config.properties"));

        Class<?> className = Class.forName(properties.getProperty("className"));
        Object obj = className.newInstance();
        Method methodName = className.getDeclaredMethod(properties.getProperty("methodName"));
        methodName.invoke(obj);
    }
}

9.通过反射越过泛型检查

public class MyTest {
    public static void main(String[] args) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
        //泛型,只在编译期有效,在运行期就擦除了。
        ArrayList<Integer> list = new ArrayList<>();
        list.add(200);
        list.add(30);
        //list.add("abc");
        //通过反射,在运行期,进行添加
        Class<? extends ArrayList> aClass = list.getClass();
        Method add = aClass.getMethod("add", Object.class);
        add.invoke(list,"abc");
        System.out.println(list);
    }
}

10.通过反射写一个通用的设置某个对象的某个属性为指定的值

public class MyTest {
    public static void main(String[] args) throws Exception {
        //通过反射写一个通用的设置某个对象的某个属性为指定的值
        Student student = new Student();
        MyUtils.setProperty(student,"name","张三");
        Object name = MyUtils.getProperty(student, "name");
        System.out.println(name);
    }
}
public class MyUtils {
    /**
     *
     * @param obj 对象
     * @param propertyName 成员变量名称
     * @param value 值
     */
    public static void setProperty(Object obj, String propertyName, Object value) throws Exception {
        Class<?> aClass = obj.getClass();
        Field field = aClass.getDeclaredField(propertyName);
        field.setAccessible(true);
        field.set(obj,value);
    }

    public static Object getProperty(Object obj, String propertyName) throws Exception {
        Class<?> aClass = obj.getClass();
        Field field = aClass.getDeclaredField(propertyName);
        field.setAccessible(true);
        Object o = field.get(obj);
        return o;
    }
}

11.动态代理

代理:本来应该自己做的事情,却请了别人来做,被请的人就是代理对象。
	举例:春季回家买票让人代买
	动态代理:在程序运行过程中产生的这个对象
	而程序运行过程中产生对象其实就是我们刚才反射讲解的内容,所以,动态代理其实就是通过反射来生成一个代理
	
	在Java中java.lang.reflect包下提供了一个Proxy类和一个InvocationHandler接口,
	通过使用这个类和接口就可以生成动态代理对象。JDK提供的代理只能针对接口做代理。
	我们有更强大的代理cglib,Proxy类中的方法创建动态代理类对象
		public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)
	最终会调用InvocationHandler的方法
		InvocationHandler Object invoke(Object proxy,Method method,Object[] args)

我们可以通过Proxy类中的静态方法获取一个代理对象:
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)

loader: 类加载器
interfaces: 接口对应的一个Class数组
InvocationHandler: 这个其实就是要代理对象所做的事情的一个类的封装
注意:JDK给我们提供的动态代理,只能对接口进行代理.

动态代理:
         *  特点:字节码随用随创建,随用随加载
         *  作用:不修改源码的基础上对方法增强
         *  分类:
         *      基于接口的动态代理
         *      基于子类的动态代理
         *  基于接口的动态代理:
         *      涉及的类:Proxy
         *      提供者:JDK官方
         *  如何创建代理对象:
         *      使用Proxy类中的newProxyInstance方法
         *  创建代理对象的要求:
         *      被代理类最少实现一个接口,如果没有则不能使用
         *  newProxyInstance方法的参数:
         *      ClassLoader:类加载器
         *          它是用于加载代理对象字节码的。和被代理对象使用相同的类加载器。固定写法。
         *      Class[]:字节码数组 代理对象需要实现的接口
         *          它是用于让代理对象和被代理对象有相同方法。固定写法。
         *      InvocationHandler:用于提供增强的代码
         *          它是让我们写如何代理。我们一般都是些一个该接口的实现类,通常情况下都是匿名内部类,但不是必须的。
         *          此接口的实现类都是谁用谁写。
new InvocationHandler() {
                    /**
                     * 作用:执行被代理对象的任何接口方法都会经过该方法
                     * 方法参数的含义
                     * @param proxy   代理对象的引用
                     * @param method  当前执行的方法
                     * @param args    当前执行方法所需的参数
                     * @return        和被代理对象方法有相同的返回值
                     */
                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                    }
                }
     - InvocationHandler 方法调用的实际处理者,代理对象的方法调用都会转发到这里。
Proxy.newProxyInstance()会返回一个实现了指定接口的代理对象,对该对象的所有方法调用都会转发给InvocationHandler.invoke()方法。 

动态代理神奇的地方就是:
1.代理对象是在程序运行时产生的,而不是编译期;
2.对代理对象的所有接口方法调用都会转发到InvocationHandler.invoke()方法,在invoke()方法里我们可以加入任何逻辑,比如修改方法参数,加入日志功能、安全检查功能等;之后我们通过某种方式执行真正的方法体

12.枚举

定义枚举类要用关键字enum
	所有枚举类都是Enum的子类,但是不要显式的写出来
	枚举类的第一行上必须是枚举项,最后一个枚举项后的分号是可以省略的,但是如果枚举类有其他的东西,这个分号就不能省略。建议不要省略
	枚举类可以有构造器,但必须是private的,它默认的也是private的。枚举项的用法比较特殊:枚举(“”);
	枚举类也可以有抽象方法,但是枚举项必须重写该方法
	枚举在switch语句中的使用
枚举类的常见方法
	int ordinal()  返回枚举项的序号
	int compareTo(E o) 比较两个枚举项的 返回的是两个枚举项序号的 差值
	String name() 获取枚举项的名称
	String toString()获取枚举项的名称
	<T> T valueOf(Class<T> type,String name) 用来获取指定的枚举项  参数1:枚举类对应的字节码对象 参数2 枚举项的名称
	values()  获取所有的枚举项
	此方法虽然在JDK文档中查找不到,但每个枚举类都具有该方法,它遍历枚举类的所有枚举值非常方便
public enum Season {
    //枚举项,多个枚举项使用逗号隔开,枚举项要位于第一行
    SPRING, SUMMER, AUTUMN, WINTER;
    //枚举类中的构造方法是私有的
    private Season(){}
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Geek Li

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值