实现较灵活的jdk动态代理

本文介绍了一种使用JDK动态代理实现目标接口方法的过程。通过自定义动态代理工厂,能够灵活地在目标方法执行前后加入自定义逻辑。文章提供了完整的代码示例,包括目标接口、实现类、切入类及测试类。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


最近准备用socket和jdk动态代理实现最简单的RPC远程调用,这里先将我写的动态代理贴出来,欢迎批评指正~

目标接口
package testProxy;

public interface TargetInterface {
    void targetMehtod(String param);
}
被代理目标实现类
package testProxy;

import org.junit.Test;

public class Target implements TargetInterface{

    @Test
    public void targetMehtod(String param){

        System.out.println("被代理的目标方法。参数为:");
    }
}

切入类
package testProxy;

public class MyAspect {

    public void before(String strParam,int intParam){
       System.out.println("有参before方法执行.参数1为:" + strParam + ",参数2为:" + intParam);
    }
    public void before(){
       System.out.println("无参before法执行");
    }

    public void after(String param){
       System.out.println("有参after方法执行.参数为:" + param);
    }
    public void after(){
       System.out.println("无参after方法执行.");
    }
}

动态代理工厂
package testProxy;

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

/**
 * 自定义灵活的动态代理
 */
public class MyReflictProxy {
    /**
     * 得到动态代理对象
     * @param target 目标对象
     * @param targetMethodName 目标对象方法名
     * @param aspect 定义切入方法的类
     * @param apectBeforeMethod 执行前插入的方法
     * @param beforeParams 执行前方法传入参数
     * @param apectAfterMethod 执行后插入的方法
     * @param afterParams 执行后方法传入参数
     * @return 代理对象
     */
    public static Object getProxyObject(final Object target, final String targetMethodName,
                                        final Object aspect, final String apectBeforeMethod,
                                        final Object[] beforeParams, final String apectAfterMethod,
                                        final Object[] afterParams){
        return Proxy.newProxyInstance(ProxyFactory.class.getClassLoader(),
                target.getClass().getInterfaces(), new InvocationHandler() {
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        Method aspectBefore;
                        Method aspectAfter;
                        //无参
                        if(null == beforeParams){
                            aspectBefore = aspect.getClass().getDeclaredMethod(apectBeforeMethod);
                            aspectBefore.invoke(aspect);
                        }else{
                            //存放所有参数的class
                            Class[] allBeforeParamClasses = new Class[beforeParams.length];
                            int i = 0;
                            //获取所有参数的class
                            for (Object param : beforeParams){
                                //若为Integer类型则转为int.class
                                if(param instanceof Integer){
                                    allBeforeParamClasses[i] = int.class;
                                }else if(param instanceof Long){
                                    allBeforeParamClasses[i] = long.class;
                                }else{
                                    allBeforeParamClasses[i] = param.getClass();
                                }
                                i++;
                            }
                            aspectBefore = aspect.getClass().getDeclaredMethod(apectBeforeMethod,allBeforeParamClasses);
                            aspectBefore.invoke(aspect,beforeParams);
                        }
                        Object back = method.invoke(target, args);
                        if(null == afterParams){
                            aspectAfter = aspect.getClass().getDeclaredMethod(apectAfterMethod);
                            aspectAfter.invoke(aspect);
                        }else{
                            Class[] allAfterParamClasses = new Class[afterParams.length];
                            int i = 0;
                            for (Object param : afterParams){
                                if(param instanceof Integer){
                                    allAfterParamClasses[i] = int.class;
                                }else if(param instanceof Long){
                                    allAfterParamClasses[i] = long.class;
                                }else{
                                    allAfterParamClasses[i] = param.getClass();
                                }
                                i++;
                            }
                            aspectAfter = aspect.getClass().getDeclaredMethod(apectAfterMethod,allAfterParamClasses);
                            aspectAfter.invoke(aspect,afterParams);
                        }
                        return back;
                    }
                });
    }
}

测试类
package testProxy;

import org.junit.Test;

/**
 * 动态代理测试
 */
public class Run {
    @Test
    public void testMyProxy() {
        TargetInterface testTarget1 = (TargetInterface) MyReflictProxy.getProxyObject(new Target(),
                "targetMehtod",new MyAspect(),"before",
                new Object[]{"有参beforeMehtod",1},"after",new Object[]{"有参beforeMehtod"});
        TargetInterface testTarget2 = (TargetInterface) MyReflictProxy.getProxyObject(new Target(),
                "targetMehtod",new MyAspect(),"before",
                null,"after",null);
        testTarget1.targetMehtod("targetMehtod");
        System.out.println();
        testTarget2.targetMehtod("targetMehtod");
    }

}

运行结果如下

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值