百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术分类 > 正文

啥?你还不懂Java反射机制与动态代理

ztj100 2025-02-15 18:23 14 浏览 0 评论

Java开发中反射和动态代理放有一定的相关性,但单纯的说动态代理是由反射机制实现的,其实是不够全面不准确的,动态代理是一种功能行为,而它的实现方法有很多。

一、反射

反射机制是Java的核心特性之一,它允许在程序运行时检查类的结构并对其进行操作。这种动态特性为程序提供了极高的灵活性,可以在运行时加载类、调用方法、修改字段值,甚至实例化对象。反射机制被广泛应用于各种框架和库中,如Spring、Hibernate、MyBatis等,它们通过反射实现了高度的动态性和灵活性。

通过反射,开发者可以:

  • 动态加载类及其依赖
  • 动态调用方法,而无需在编译时知道具体方法
  • 动态修改对象的字段,甚至是私有字段

尽管反射为开发者提供了强大的功能,但也伴随着一定的性能开销和安全隐患,因此在实际使用中需要谨慎。

1. 实现方式

想象你是一名厨师,而食谱就是一个类。食谱中记录了所有菜品的制作方法,而你作为厨师通过食谱(类)指导自己的烹饪过程。这就类似于反射机制,你可以在程序运行时查看类的结构并通过类中的方法制作“菜品”。

// 示例:获取类信息和方法
Class recipeClass = Class.forName("com.example.Recipe");
String recipeName = recipeClass.getName();
System.out.println("菜品名称: " + recipeName);


// 动态调用方法
Method cookMethod = recipeClass.getDeclaredMethod("cook");
cookMethod.invoke(recipeObject);  


2. 获取类(Class)对象

在使用反射机制时,第一步是获取Class对象。Java为此提供了三种方式:

1.通过forName() -> 示例:Class.forName(“People”)
2.通过getClass() -> 示例:new People().getClass()
3.class直接获取 -> 示例:PeopleImpl.class

2.1静态方法调用

使用 getMethod(xx) 获取到对应的方法,直接使用 invoke(xx)就可以了。

public static void main(String[] args) {
    Class myClass = Class.forName("example.People");
    // 调用静态(static)方法
    Method getSex = myClass.getMethod("getName");
    getSex.invoke(myClass);
}

2.2 普通方法调用

使用getMethod() 获取方法,可以声明需要传递的参数的类型。

Object object = myClass.newInstance();
Method method = myClass.getMethod("hello",String.class);
method.invoke(object,"word!");

2.3 调用私有方法

调用私有方法的关键是设置 setAccessible(true) 属性,修改访问限制,这样设置之后就可以进行调用

Method greetMethod = clazz.getDeclaredMethod("greet", String.class);
greetMethod.setAccessible(true); // 允许访问私有方法
greetMethod.invoke(personObj, "John Doe");  // 调用方法

3. 应用

反射机制在实际开发中有广泛的应用,尤其是在框架和工具开发中。以下是一些常见的应用场景:

  • 动态加载类与方法例如Spring框架通过反射来动态加载Bean对象并调用其初始化方法。
  • 序列化与反序列化例如JSON库(如Jackson和Gson)通过反射将JSON字符串转换为Java对象
  • 注解处理在Spring等框架中,通过反射扫描类上的注解并执行相应的逻辑。
  • 动态代理例如Java AOP(面向切面编程)利用动态代理实现方法的拦截和增强。


二、动态代理

动态代理是一种方便运行时动态构建代理、动态处理代理方法调用的机制,很多场景都是利用类似机制做到的,比如用来包装 RPC 调用、面向切面的编程(AOP)。

实现动态代理的方式很多,比如 JDK 自身提供的动态代理,就是主要利用了上面提到的反射机制。还有其他的实现方式,比如利用传说中更高性能的字节码操作机制,类似 ASM、cglib(基于 ASM)等。

Java的动态代理主要有两种方式:

1. JDK动态代理

JDK动态代理是通过实现接口的方式生成代理类。Proxy类和InvocationHandler接口是JDK动态代理的核心。

public class CalculatorProxy implements InvocationHandler {
    private final Object target;


    public CalculatorProxy(Object target) {
        this.target = target;
    }


    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("方法 " + method.getName() + " 开始执行");
        Object result = method.invoke(target, args);
        System.out.println("方法 " + method.getName() + " 执行完毕");
        return result;
    }


    public static Object newProxyInstance(Object target) {
        return Proxy.newProxyInstance(
                target.getClass().getClassLoader(),
                target.getClass().getInterfaces(),
                new CalculatorProxy(target));
    }
}

注意:JDK Proxy 只能代理实现接口的类(即使是extends继承类也是不可以代理的)。

2. CGlib代理

JDK 动态代理机制只能代理实现了接口的类,CGlib 是针对类来实现代理的,他的原理是对指定的目标类生成一个子类,并覆盖其中方法实现增强,但因为采用的是继承,所以不能对 final 修饰的类进行代理。

CGlib 的调用通过实现 MethodInterceptor 接口的 intercept 方法,调用 invokeSuper 进行动态代理的,可以直接对普通类进行动态代理。

class CglibProxy implements MethodInterceptor {
    private Object target; // 代理对象
    public Object getInstance(Object target) {
        this.target = target;
        Enhancer enhancer = new Enhancer();
        // 设置父类为实例类
        enhancer.setSuperclass(this.target.getClass());
        // 回调方法
        enhancer.setCallback(this);
        // 创建代理对象
        return enhancer.create();
    }
    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        System.out.println("调用前");
        Object result = methodProxy.invokeSuper(o, objects); // 执行方法调用
        System.out.println("调用后");
        return result;
    }
}


public static void main(String[] args) {
    // CGLIB 动态代理调用
    CglibProxy proxy = new CglibProxy();
    Panda panda = (Panda)proxy.getInstance(new Panda());
    panda.eat();
}


3. 应用

动态代理主要用于 AOP 场景,例如日志、权限控制、事务管理等。动态代理还常用于 RPC 框架中实现远程服务调用的透明化。例如,Spring AOP 使用动态代理来实现切面编程,dubbo使用动态代理来实现远程调用。

三. 总结

Java的反射机制和动态代理为开发者提供了强大的动态功能,使得程序可以在运行时灵活地处理类、对象及其行为。虽然这些特性极大地增强了灵活性和可扩展性,但也带来了性能开销和安全隐患。因此,反射和动态代理的使用应该谨慎,尤其是在对性能有较高要求的场景下。

通过对反射机制和动态代理的深入理解,两者在实际开发中常常各自发挥作用,或者结合使用,以实现复杂的动态行为和灵活的系统设计。


如果这篇文章对您有所帮助,或者有所启发的话,帮忙给个赞 谢谢!!!

相关推荐

再说圆的面积-蒙特卡洛(蒙特卡洛方法求圆周率的matlab程序)

在微积分-圆的面积和周长(1)介绍微积分方法求解圆的面积,本文使用蒙特卡洛方法求解圆面积。...

python编程:如何使用python代码绘制出哪些常见的机器学习图像?

专栏推荐...

python创建分类器小结(pytorch分类数据集创建)

简介:分类是指利用数据的特性将其分成若干类型的过程。监督学习分类器就是用带标记的训练数据建立一个模型,然后对未知数据进行分类。...

matplotlib——绘制散点图(matplotlib散点图颜色和图例)

绘制散点图不同条件(维度)之间的内在关联关系观察数据的离散聚合程度...

python实现实时绘制数据(python如何绘制)

方法一importmatplotlib.pyplotaspltimportnumpyasnpimporttimefrommathimport*plt.ion()#...

简单学Python——matplotlib库3——绘制散点图

前面我们学习了用matplotlib绘制折线图,今天我们学习绘制散点图。其实简单的散点图与折线图的语法基本相同,只是作图函数由plot()变成了scatter()。下面就绘制一个散点图:import...

数据分析-相关性分析可视化(相关性分析数据处理)

前面介绍了相关性分析的原理、流程和常用的皮尔逊相关系数和斯皮尔曼相关系数,具体可以参考...

免费Python机器学习课程一:线性回归算法

学习线性回归的概念并从头开始在python中开发完整的线性回归算法最基本的机器学习算法必须是具有单个变量的线性回归算法。如今,可用的高级机器学习算法,库和技术如此之多,以至于线性回归似乎并不重要。但是...

用Python进行机器学习(2)之逻辑回归

前面介绍了线性回归,本次介绍的是逻辑回归。逻辑回归虽然名字里面带有“回归”两个字,但是它是一种分类算法,通常用于解决二分类问题,比如某个邮件是否是广告邮件,比如某个评价是否为正向的评价。逻辑回归也可以...

【Python机器学习系列】拟合和回归傻傻分不清?一文带你彻底搞懂

一、拟合和回归的区别拟合...

推荐2个十分好用的pandas数据探索分析神器

作者:俊欣来源:关于数据分析与可视化...

向量数据库:解锁大模型记忆的关键!选型指南+实战案例全解析

本文较长,建议点赞收藏,以免遗失。更多AI大模型应用开发学习视频及资料,尽在...

用Python进行机器学习(11)-主成分分析PCA

我们在机器学习中有时候需要处理很多个参数,但是这些参数有时候彼此之间是有着各种关系的,这个时候我们就会想:是否可以找到一种方式来降低参数的个数呢?这就是今天我们要介绍的主成分分析,英文是Princip...

神经网络基础深度解析:从感知机到反向传播

本文较长,建议点赞收藏,以免遗失。更多AI大模型应用开发学习视频及资料,尽在...

Python实现基于机器学习的RFM模型

CDA数据分析师出品作者:CDALevelⅠ持证人岗位:数据分析师行业:大数据...

取消回复欢迎 发表评论: