动态代理

本文深入解析动态代理技术,包括基于接口的JDK动态代理和基于子类的CGLIB动态代理,通过实例演示如何为现有类添加新功能,而不修改其源代码。

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

动态代理:在不改变代码的基础上为某一个类添加功能
<1>基于接口的动态代理:
被代理类最少实现一个接口,代理类要和被代理类具有相同的功能,也就是实现相同的接口,代理类要和被代理类使用相同的类加载器
Iacort proxyActor =(Iacort) Proxy.newProxyInstance(actor.getClass().getClassLoader(), actor.getClass().getInterfaces(), handler);
被代理类要与一个增强类,次增强了要实现InvocationHandler接口,
代码:
接口:
public interface Iacort {
public void play(float money);
public void dangerplay(float money);
}

被代理类:
public class Actor implements  Iacort{
    public void play(float money){
        System.out.println("拿钱基本演出" +money+"元");
    }
    public void dangerplay(float money){
        System.out.println("拿钱危险演出" +money+"元");
    }
}
增强类:
public class actorHandler implements InvocationHandler {
    //创建代理对象,准备为其添加新的功能
     Actor actor = new Actor();
     Object invoke =null;

    /**
     *此函数用来的增强被代理类的功能,没到用户调用被代理类的方法是都会来invoke函数来报道。
     * @param proxy  代理对象的引用
     * @param method  客户调用的方法
     * @param args    调用方法所需要的参数
     * @return         调用方法的返回值,void实际也是对象,是继承了object类的
     * @throws Throwable
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println(args);
        //Float money = (Float) args;
        System.out.println(args[0]+"元");
        if ("play".equals(method.getName()) && (Float)args[0]>1000){
             invoke = method.invoke(actor, args);
        }
        if ("dangerplay".equals(method.getName()) && (Float)args[0]>5000){
            invoke = method.invoke(actor, (Float)args[0]);
        }
        return invoke;
    }
}
测试:
public class Test {
    @org.junit.Test
    public void test(){
       Actor actor = new Actor();
        actorHandler handler = new actorHandler();
        /*
         * 创建代理对象
         * param  actor.getClass().getClassLoader()  这是类加载器,代理对象的类加载器要和被代理类的一样,这里固定的写法,代理谁用谁的类加载器
         * param actor.getClass().getInterfaces()   被代理类实现的接口,是个字节数组,要求代理类和被代理类要有相同的功能
         * param handler 此参数用 来增强被代理的对象的功能的
         */
        Iacort procyActor =(Iacort) Proxy.newProxyInstance(actor.getClass().getClassLoader(), actor.getClass().getInterfaces(), handler);

        System.out.println(procyActor);
        /**
         * 测试代理类*/
        procyActor.dangerplay(1000);
        procyActor.play(20000f);
    }
}

<2>基于子类(继承)的动态代理:cglib实现
/**
* cglib实现动态代理
@param actor.getClass()被代理类的字节码
@param new MethodInterceptor()具体怎么代理 就是jdk实现中InvactionHandler接口
/
Iacort o = (Iacort) Enhancer.create(actor.getClass(), new MethodInterceptor() {
/

*
* @param o 代理类
* @param method 客户调用的方法
* @param objects 执行当前方法需要的参数
* @param methodProxy 当前执行方法的代理方法,一般不会使用
* @return
* @throws Throwable
*/
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
Float money = (Float)objects[0];
Object res = null ;
if(money>10000 && “play”.equals(method.getName())){
res = method.invoke(actor, money);
}
if(money>50000 && “dangerplay”.equals(method.getName())){
res = method.invoke(actor, money);
}
return res;
}
});

资源下载链接为: https://pan.quark.cn/s/f989b9092fc5 今天给大家分享一个关于C#自定义字符串替换方法的实例,希望能对大家有所帮助。具体介绍如下: 之前我遇到了一个算法题,题目要求将一个字符串中的某些片段替换为指定的新字符串片段。例如,对于源字符串“abcdeabcdfbcdefg”,需要将其中的“cde”替换为“12345”,最终得到的结果字符串是“ab12345abcdfb12345fg”,即从“abcdeabcdfbcdefg”变为“ab12345abcdfb12345fg”。 经过分析,我发现不能直接使用C#自带的string.Replace方法来实现这个功能。于是,我决定自定义一个方法来完成这个任务。这个方法的参数包括:原始字符串originalString、需要被替换的字符串片段strToBeReplaced以及用于替换的新字符串片段newString。 在实现过程中,我首先遍历原始字符串,查找需要被替换的字符串片段strToBeReplaced出现的位置。找到后,就将其替换为新字符串片段newString。需要注意的是,在替换过程中,要确保替换操作不会影响后续的查找和替换,避免遗漏或重复替换的情况发生。 以下是实现代码的大概逻辑: 初始化一个空的字符串result,用于存储最终替换后的结果。 使用IndexOf方法在原始字符串中查找strToBeReplaced的位置。 如果找到了,就将originalString中从开头到strToBeReplaced出现位置之前的部分,以及newString拼接到result中,然后将originalString的查找范围更新为strToBeReplaced之后的部分。 如果没有找到,就直接将剩余的originalString拼接到result中。 重复上述步骤,直到originalStr
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值