java动态代理

动态代理

参考Java动态代理机制详解(JDK 和CGLIB,Javassist,ASM)

代理的实质是在运行期间手动创建class类,对被代理对象的方法进行代理,调用被代理对象的方法,动态代理就是动态的创建Proxy对象,用完之后销毁class类,避免冗杂,动态代理的实现方式主要有以下两种。
CGLIB代理主要是对指定的类生成一个子类,覆盖其中的所有方法,所以该类或方法不能声明称final的。

一、通过jdk实现InvocationHandler实现动态代理
1、定义接口
package cn.zlz.proxy.jdk;

public interface IComputorService {

    /**
     * 卖电脑
     * @param brand
     */
    public void sellComputor(String brand);
    /**
     * 修电脑
     */
    public void repairComputor(String brand);

}
2、定义接口实现类
package cn.zlz.proxy.jdk;

public class ThinkPadSeller implements IComputorService{

    public void sellComputor(String brand) {
        System.out.println("sell the thinkPad computor");

    }

    public void repairComputor(String brand) {
        System.out.println("repair the thinkPad computor");
    }

}
3、定义生成代理对象
package cn.zlz.proxy.jdk;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

public class SimpleProxyImpl implements InvocationHandler {

    // 被代理对象
    private ThinkPadSeller thinkPadSeller;

    public SimpleProxyImpl(ThinkPadSeller thinkPadSeller) {
        super();
        this.thinkPadSeller = thinkPadSeller;
    }

    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("代理开始");
        // 调用被代理对象
        method.invoke(thinkPadSeller, args);
        System.out.println("代理结束");
        return null;
    }
    //提供方法获取代理对象  
    public IComputorService newProxy(){  

        //使用Proxy类创建代理对象  
        IComputorService proxyInstance = (IComputorService) Proxy.newProxyInstance(thinkPadSeller.getClass().getClassLoader(), //使用被代理对象的加载器  
                thinkPadSeller.getClass().getInterfaces(), //使用被代理对象的接口  
                this );//匿名内部类比较坑,所以我们找一个类实现并覆写方法,直接用本类,现成的..  
        return proxyInstance;  
    }  
}
4、main函数测试
package cn.zlz.proxy.jdk;

import java.lang.reflect.Proxy;

/**
 *  通过jdk的实现invocationHandler接口只能代理实现接口的对象
 *  为了解决这个问题,就有了动态地创建Proxy的想法:在运行状态中,需要代理的地方,根据接口 和被代理对象,
 *  动态地创建一个Proxy,用完之后,就会销毁,这样就可以避免了Proxy 角色的class在系统中冗杂的问题了。
 *
 */
public class Main {

    public static void main(String[] args) {
        /**
         * 使用Proxy创建代理对象
         * 1、被代理对象
         * 2、被代理对象实现的接口s
         * 3、Invocation实现对象
         */
        //使用Proxy类创建代理对象
        Class beProxyClazz = ThinkPadSeller.class;
        ThinkPadSeller thinkPadSeller = new ThinkPadSeller();
        ClassLoader classLoader = beProxyClazz.getClassLoader();
        Class[] interfaces = beProxyClazz.getInterfaces();
        SimpleProxyImpl simpleProxyImpl = new SimpleProxyImpl(thinkPadSeller);
        // 根据上面提供的信息,创建代理对象 在这个过程中,JDK会通过根据传入的参数信息动态地在内存中创建和.class 文件等同的字节码 ,然后根据相应的字节码转换成对应的class,然后调用newInstance()创建实例 
        IComputorService proxy = (IComputorService) Proxy.newProxyInstance(classLoader, interfaces, simpleProxyImpl);
        /*
         * 生成的代理对象编译后的代码为 public final repairComputor(){this.h.invoke(this, m3, null);m3 = Class.forName("cn.zlz.proxy.jdk.ThinkPadSeller").getMethod("repairComputor", [String.class]); }
         * this指的是invocation的实现类,调用invoke方法,并将被代理对象的方法作为参数传递
         */
        proxy.repairComputor("thinkPad");
    }
}
而、通过cglib实现动态代理
1、定义被代理对象
package cn.zlz.proxy.cglib;

public class ThinkPadSeller {

    public void sellComputor(String brand) {
        System.out.println("sell the thinkPad computor");

    }

    public void repairComputor(String brand) {
        System.out.println("repair the thinkPad computor");
    }

}
2、实现cglib的MethodInterceptor
package cn.zlz.proxy.cglib;

import java.lang.reflect.Method;

import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

/*
 * cglib代理,实现MethodInterceptor
 */
public class CglibProxy implements MethodInterceptor {

    public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
        System.out.println("开始代理");
        //代理类是继承的被代理类,调用父类的原方法
        proxy.invokeSuper(obj, args);  

        System.out.println("结束代理");
        return null;
    }

}
3、main函数调试
package cn.zlz.proxy.cglib;

import net.sf.cglib.proxy.Enhancer;

/*
 * 代理对象继承被代理对象
  1.查找被代理类的所有非final 的public类型的方法定义;
  2.将这些方法的定义转换成字节码;
  3.将组成的字节码转换成相应的代理的class对象;
  4.实现 MethodInterceptor接口,用来处理 对代理类上所有方法的请求(这个接口和JDK动态代理InvocationHandler的功能和角色是一样的)
 */
public class Main {
    public static void main(String[] args) {
        CglibProxy cglibProxy = new CglibProxy();
        // cglib 中加强器,用来创建动态代理
        Enhancer enhancer = new Enhancer();
        // 设置要创建动态代理的类,即父类
        enhancer.setSuperclass(ThinkPadSeller.class);
        // 设置回调,这里相当于是对于代理类上所有方法的调用,都会调用CallBack,而Callback则需要实行intercept()方法进行拦截
        enhancer.setCallback(cglibProxy);
        ThinkPadSeller proxy = (ThinkPadSeller) enhancer.create();
        proxy.repairComputor("thinkpad");
    }
}
### Java 动态代理的实现原理与用法 #### 1. Java 动态代理概述 Java 动态代理是一种在运行时动态生成代理对象的技术,它允许开发者无需提前定义具体的代理类即可完成方法拦截和增强功能。这种机制广泛应用于 AOP(面向切面编程)、事务管理以及日志记录等领域[^2]。 #### 2. 动态代理的核心组件 动态代理主要依赖于 `java.lang.reflect.Proxy` 类和 `InvocationHandler` 接口来实现。以下是其核心组成部分: - **Proxy 类**: 提供了用于创建动态代理实例的方法。 - **InvocationHandler 接口**: 定义了一个处理方法调用的回调接口,通过该接口可以自定义代理行为。 当客户端调用代理对象上的某个方法时,实际执行的是由 InvocationHandler 处理逻辑所指定的操作[^3]。 #### 3. JDK 原生动态代理实现 JDK 动态代理基于反射技术,在运行期间为一组接口动态生成代理类及其实例。具体流程如下: - 创建一个实现了 `InvocationHandler` 接口的对象。 - 使用 `Proxy.newProxyInstance()` 方法传入目标类加载器、目标类实现的一组接口列表以及上述 handler 对象,从而获得代理实例。 下面是一个简单的示例代码展示如何利用 JDK 动态代理实现基本的日志打印功能: ```java import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; // 定义服务接口 interface GreetingService { void sayHello(String name); } // 实现服务接口的具体业务逻辑 class SimpleGreetingServiceImpl implements GreetingService { @Override public void sayHello(String name) { System.out.println("Hello, " + name); } } // 自定义 InvocationHandler 来拦截并扩展方法调用 class LoggingInvocationHandler implements InvocationHandler { private Object target; // 被代理的目标对象 public LoggingInvocationHandler(Object target) { this.target = target; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("[Before] Executing: " + method.getName()); // 执行原始方法 Object result = method.invoke(target, args); System.out.println("[After] Finished execution."); return result; } } public class JdkDynamicProxyDemo { public static void main(String[] args) { // 初始化真实的服务实现 GreetingService greetingService = new SimpleGreetingServiceImpl(); // 构建带有日志功能的代理对象 GreetingService proxyInstance = (GreetingService) Proxy.newProxyInstance( greetingService.getClass().getClassLoader(), greetingService.getClass().getInterfaces(), new LoggingInvocationHandler(greetingService)); // 测试代理效果 proxyInstance.sayHello("World"); } } ``` 此程序展示了如何通过动态代理增加额外的功能而不修改原有代码结构[^1]。 #### 4. CGLib 动态代理实现 除了 JDK 内置支持外,还可以借助第三方库如 CGLib 进行动态代理操作。CGLib 主要针对那些未继承任何接口或者希望直接操控实体类场景下更为适用。它的内部工作机理涉及字节码层面操作,能够子类化任意非 final 的普通类,并重写其中的方法达到相同目的。 需要注意的是,由于两者设计初衷不同,在性能表现上可能会有所差异;通常情况下,如果仅需对接口做封装,则推荐优先考虑更轻量级也更加直观易懂的 JDK 方案。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值