首页 > 其他分享 >反射、代理简单理解

反射、代理简单理解

时间:2024-10-25 21:47:22浏览次数:9  
标签:反射 对象 代理 class 获取 理解 方法 Class

反射

反射允许对成员变量,成员方法和构造方法的信息进行编程访问

但是获取不是从类里面获取的,是从类的字节码(.class)文件中获取的,所以我们首先要学习如何获取类的class对象。

在Java中,定义好了一个类Class,就是用来描述字节码文件的

获取class对象的三种方式   

1. Class.forname("全类名");

2.类名.class

3.对象.getClass();

这三种方式对应编码的三个阶段:

代码实现,Student类是一个标准的JavaBean,下述代码是获取Student类的class对象的三种方式

public class MyReflectDemo1 {
    public static void main(String[] args) throws ClassNotFoundException {
        /*
        获取class对象的三种方式
        1.Class.forName("全类名")
        2.类名.class
        3.对象.getClass();
         */
        //1.第一种方式
        //最为常用
        //全类名:包名+类名
        Class clazz = Class.forName("myreflect1.Student");

        //2.第二种方式
        //一般更多当参数使用
        Class clazz2 = Student.class;

        //3.第三种方式
        //当我们已经有该类对象才可以使用
        Student s = new Student();
        Class clazz3 = s.getClass();
        
        //打印
        System.out.println(clazz);
        System.out.println(clazz2);
        System.out.println(clazz3);
    }
}

在Java中一切皆对象。

class对象,构造方法,成员变量,成员方法。都有与之相对应的类,以便获取对象,如图示:

                                ​​​​​​​        ​​​​​​​        ​​​​​​​        ​​​​​​​        

利用反射获取构造方法

Class类中用于获取构造方法的方法
    Constructor<?>[] getConstructors():返回所有公共构造方法对象的数组
    Constructor<?>[] getDeclaredConstructors():返回所有构造方法对象的数组
    Constructor<T> getConstructor(class<?>... parameterTypes):返回单个公共构造方法对象
    Constructor<T> getDeclaredConstructor(class<?>...parameterTypes):返回单个构造方法对象

    Constructor类中用于创建对象的方法
    T newInstance(object... initargs):根据指定的构造方法创建对象
    setAccessible(boolean flag):设置为true,表示取消访问检查

其中,在用private修饰符的构造方法创建对象时,会报错。此时需要用暴力反射。

con4.setAccessible(true);   表示临时取消权限的校验

利用反射获取成员变量

Class类中用于获取成员变量的方法
    Field[] getFields():                    返回所有公共成员变量对象的数组
    Field[] getDeclaredFields():            返回所有成员变量对象的数组
    Field getField(String name):            返回单个公共成员变量对象
    Field getDeclaredField(String name):    返回单个成员变量对象

 Field类中用于创建对象的方法
    void set(Object obj, Object value):     赋值
    Object get(Object obj)                  获取值

利用反射获取成员方法

Class类中用于获取成员方法的方法
Method[] getMethods():返回所有公共成员方法对象的数组,包括继承的
Method[] getDeclaredMethods():返回所有成员方法对象的数组,不包括继承的
Method getMethod(String name, Class<?>.. parameterTypes):返回单个公共成员方法对象
Method getDeclaredMethod(String name, Class<?>... parameterTypes):返回单个成员方法对象

获取方法的修饰符
获取方法的名字
获取方法的形参
获取方法的返回值
获取方法抛出的异常

获取方法后,如何将方法运行起来呢?

Method类中用于创建对象的方法
Object invoke(Object obj, Object... args):运行方法
参数一:用obj对象调用该方法
参数二:调用方法的传递的参数(如果没有就不写)
返回值:方法的返回值(如果没有就不写)

        Student s = new Student();
        m.setAccessible(true);

        //参数1 s:表示方法的调用者
        //参数2 "汉堡包":表示在调用方法的时候传递的实际参数
        String result = (String) m.invoke(s, "汉堡包");
        System.out.println(result);

反射的作用:

1.获取一个类里面所有的信息,获取到了之后,再执行其他的业务逻辑

2.结合配置文件,动态创建对象并调用方法

示例一:将对象所有的字段名和值,保存到文件中,首先创建两个完整的JavaBean类

        

通过IO流,写入txt文件

运行结果:

示例二:结合配置文件动态创建对象

在配置文件中写入全类名,方法名。表示程序运行时创建类对象并调用方法。

在主程序中,分为以下几个步骤:

1.读取配置文件中的信息

//1.读取配置文件中的信息
        Properties prop = new Properties();
        FileInputStream fis = new FileInputStream("prop\\properties.java");
        prop.load(fis);

        fis.close();
        System.out.println(prop);

2.获取全类名和方法名

        //2.获取全类名和方法名
        String className = (String)prop.get("classname");
        String methodName = (String)prop.get("method");

        System.out.println(className);
        System.out.println(methodName);

3.利用反射创建对象并运行方法

        //3.利用反射创建对象并运行方法
        Class clazz = Class.forName(className);

4.获取构造方法

        //4.获取构造方法
        Constructor con = clazz.getDeclaredConstructor();
        Object o = con.newInstance();
        System.out.println(o);

5.获取成员方法并运行

        //5.获取成员方法并运行
        Method method = clazz.getDeclaredMethod(methodName);
        method.setAccessible(true);
        method.invoke(o);

运行结果如下

如果要创建其他类对象,运行其中的方法,不用更改程序,只需更改配置文件

例如:创建Teacher类对象,运行其中的teach方法,在properties.java中更改为:

运行结果如下:

总结:

代理

在Java中,代理(proxy)是一种设计模式,它允许你在不改变原始类的前提下增强、控制或简化

方法调用。代理模式广泛用于面向对象编程中的 AOP(面向切面编程),尤其是在框架如 Spring

中。

代理可以简单理解为:在原始对象前插入一个“中间人”来控制对原始对象的访问。这使得你可以在

方法调用前后做一些额外的操作,如日志记录、权限检查、事务管理等。

代理长什么样?

代理里面就是对象要被代理的方法

Java通过什么来保证代理的样子?

通过接口保证,对象和代理需要实现同一个接口,接口中就是被代理的所有方法

举例:

坤哥会唱歌,会跳舞,所以坤哥类里面有这两种方法,被代理的对象(坤哥)有几种方法,代理就需要几种方法与之对应,坤哥和代理的方法都要统一定义在接口中

代理的分类:

1、静态代理(Static Proxy):代理类在编译时已经存在,开发者需要手动编写代理类

2、动态代理(Dynamic Proxy):在运行时动态创建的代理类

静态代理

动态代理

以坤哥的唱歌,跳舞的需求为例,

步骤1:定义一个Star接口,接口定义唱歌和跳舞的抽象方法,其次定义一个BigStar类,生成标准JavaBean,实现Star接口,重写里面的抽象方法,如图:

        

步骤2:创建一个代理类,在代理类中创建实现代理的方法,创建代理需要传入被代理的对象,即为BigStar类的对象,返回的是代理对象,由于代理对象也需要实现Star接口,所以这边直接可以写Star(接口多态)。

/**
     * 方法的作用:
     * 给一个明星的对象,创建一个代理
     * @param bigStar:被代理的明星对象
     * @返回值:给明星创建的代理
     */
public static Star createProxy(BigStar bigStar) {
}

步骤3:完善创建代理的方法,静态类Proxy中提供了为对象产生代理对象的方法,因此直接调用方法,返回的是代理对象。

方法中传递的三个参数的简单理解:

ClassLoader loader:用于加载生成的代理类的类加载器。一般传递目标类(即代理对象所属类)的类加载器。当前类ProxyUtil就是代理类,故传递当前类的类加载器ProxyUtil.class.getClassLoader()

class<?>[ ] interfaces:指定代理类要实现的接口。数组决定了代理对象的类型和方法列表。如果有多个接口需要实现,都需要写入{ }。此处代理类要实现的接口为Star,故传入Star字节码文件的Class对象

InvocationHandler h:一个接口实例,用于定义代理对象的行为。通过匿名内部类实现InvocationHandler接口,并重写invoke方法。


需求:想要明星唱首歌

1.获取代理的对象

代理对象 = ProxyUtil.createProxy(大明星的对象);

2.再调用代理的唱歌方法

代理对象.唱歌的方法("歌曲名称");

当外界代理对象调用唱歌方法以后,唱歌方法会去调用invoke方法。

InvocationHandler中的invoke方法:

方法定义了代理对象在调用方法时的行为。invoke方法参数解释:

object proxy:代理对象本身,一般不用操作这个对象

Method method:表示正常被调用的方法对象

object[ ] args:表示调用方法时的参数

写一个测试方法进行测试

运行结果如下:

标签:反射,对象,代理,class,获取,理解,方法,Class
From: https://blog.csdn.net/m0_58087746/article/details/143224098

相关文章

  • CF605E Intergalaxy Trips 与 对期望的进一步理解
    简化题面给一张无向图,在每一时刻,每一条边权值都为\(1\),出现的概率都是给定的(但不完全相同),问最优决策下\(1\)到\(n\)的期望。Attention:是每条边都会有概率出现,而不是走每条边都会有概率成功,这就意味着,我在某一点的不同的边的出现的情况下,我会做出选择。#sol.定义......
  • [超详细有案例]理解白盒测试的5种逻辑覆盖
    [超详细有案例]理解白盒测试的5种逻辑覆盖    白盒测试是穷举路径测试,在逻辑覆盖中有6种,分别是语句覆盖,判定覆盖,条件覆盖,判定/条件覆盖,组合覆盖,路径覆盖,下面我将以每种覆盖的定义,实例讲解,优点,缺点了帮助大家理解。(1)语句覆盖        语句覆盖是最起码的结构覆......
  • Docker | 初次认识Docker并理解Docker中的镜像、容器、仓库概念
    认识Docker1.Docker简介1.1是什么1.2容器与虚拟机比较传统虚拟机技术容器虚拟化技术对比容器和虚拟机有什么不同?1.3能干嘛1.4安装⭐1.5Docker的基本组成⭐⭐Docker平台架构图解(入门版)Docker工作原理Docker平台架构图解架构版(深入版)1.Docker简介1.1是......
  • Java的反射到底是有什么用处
    Java的反射提供了一种强大的机制,允许程序在运行时检查、修改类的行为。这一特性在以下多个方面发挥着重要作用:1.动态创建对象和调用方法;2.实现通用代码;3.支持各种框架和库;4.进行测试和调试。反射使得可以在运行时创建任意类的对象,调用任意方法,这对于编写通用代码或框架非常有......
  • 【Python爬虫实战】深入理解Python异步编程:从协程基础到高效爬虫实现
    #1024程序员节|征文#  ......
  • 深入理解偏向锁、轻量级锁、重量级锁
    一、对象结构和锁状态synchronized关键字是java中的内置锁实现,内置锁实际上就是个任意对象,其内存结构如下图所示其中,MarkWord字段在64位虚拟机下占64bit长度,其结构如下所示可以看到MarkWord字段有个很重要的作用就是记录当前对象锁状态,最后3bit字段用来标记当前锁状态是无......
  • 理解OAuth2与用户账户与授权UAA的关系
    目录关于OAuth2OAuth2的核心组件授权流程授权模式使用场景优点与缺点关于UAA技术解释UAA与OAuth2的关系关于OAuth2‌‌OAuth2(开放授权2.0)是一个开放标准,用于授权第三方应用程序访问用户资源,而无需共享用户的用户名和密码。‌OAuth2通过颁发令牌来实现授权,提高了系......
  • IIS使用反向代理,解决路径包含特殊字符无法访问的问题
    环境:操作系统:WindowsServer2019IIS版本:10问题试用Nocobase的时候,遇到400BadRequest的报错。 直接访问的话,报错页面是非常常见的RuntimeError。诡异这个问题,在开发端(Windows11)的IIS中不会出现。同样的IIS版本。解决办法先说结论,时间比较赶的朋友直接试这个就可以......
  • 什么是云访问安全代理(CASB)
    云访问安全代理(CASB)是一种位于企业和云服务提供商之间的安全工具,用于监控、控制和保护企业在云环境中的数据和应用。CASB可以作为中间层,帮助企业实现对云服务的可见性、数据安全性、合规性和访问控制等管理功能。一、CASB概述云计算和云服务的广泛应用使得企业和组织能够以更......
  • 手机录屏技术原理解析与应用场景
    手机录屏技术的原理主要依赖于设备操作系统的功能以及硬件支持,以下是详细步骤:1.屏幕图像捕获手机录屏的核心在于捕获屏幕上的图像。现代智能手机操作系统(如Android和iOS)通过系统接口将当前显示的内容逐帧捕获,并传递给录屏应用。这些系统API会捕获屏幕上的像素数据,然后将这些数据转......