多学习。

动态代理

概述

特点

字节码随用随创建,随用随加载。

作用

不修改源码的基础上对方法增强

分类

1.基于接口的动态代理
————涉及的类:Proxy
————提供者: JDK官方
————如何创建代理对象:使用Proxy类中的newProxyInstance方法
————创建代理对象的要求:被代理类至少实现一个接口,如果没有则不能使用
————newProxyInstance方法的参数:
——————————————————ClassLoader: 类加载器,用于加载代理对象字节码的。和被代理对象使用相同的类加载器(固定写法)
——————————————————Class[]: 字节码数组,用于让代理对象与被代理对象有相同的方法
——————————————————InvocationHandler: 用于提供增强代码。一般都是写一个该接口的实现类,通常都是匿名内部类,但不是必须的。

2.基于子类的动态代理
————涉及的类:Enhancer
————提供者:第三方cglib库
————如何创建代理对象:使用Enhancer类中的create方法
————创建代理对象的要求:被代理类不能是最终类(原因:(final)最终类不能有子类)
————create方法的参数:
——————————————————Class:字节码。用于被代理对象的字节码
——————————————————Callback:用于提供增强的代码,也是用于方法的拦截。一般都是写一个该接口的实现类,通常都是匿名内部类,但不是必须的。我们一般写的都是该接口的子接口实现类:MethodInterceptor

基于接口的动态代理实例

生产商接口

package com.czy.proxy;

public interface IProducer {
    /**
     * 销售
     * @param money
     */
    public void sellProduct(float money);

    /**
     * 售后
     * @param money
     */
    public void afterService(float money);
}

生产商实现类

package com.czy.proxy;

/**
 * 生产者
 */

public class Producer implements IProducer{

    /**
     * 销售
     * @param money
     */
    public void sellProduct(float money){
        System.out.println("销售产品,并拿到钱:"+money);
    }

    /**
     * 售后
     * @param money
     */
    public void afterService(float money){
        System.out.println("提供售后服务,并拿到钱:"+money);
    }


}

客户通过代理商买物品

package com.czy.proxy;

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

/**
 * 模拟一个消费者
 */
public class Client {
    public static void main(String[] args) {
        final Producer producer = new Producer();
        IProducer proxyProducer = (IProducer) Proxy.newProxyInstance(producer.getClass().getClassLoader(), producer.getClass().getInterfaces(), new InvocationHandler() {
            /**
             * 作用:执行被代理对象的任何接口方法都会经过该方法
             * @param proxy 代理对象的引用
             * @param method    当前执行的方法
             * @param args  当前执行方法所需的参数
             * @return  和被代理对象有相同的返回值
             * @throws Throwable
             */
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                //提供增强代码
                Float money = (Float)args[0];
                if("sellProduct".equals(method.getName())){
                    //代理商吃了二成回扣
                    return method.invoke(producer,money*0.8f);
                }
                return method.invoke(producer,args);
            }
        });
        proxyProducer.sellProduct(10000f);
    }
}

基于子类实现的动态代理

依赖

        <dependency>
            <groupId>cglib</groupId>
            <artifactId>cglib</artifactId>
            <version>2.1_3</version>
        </dependency>

生产商类(无继承任何接口)

package com.czy.cglib;

import com.czy.proxy.IProducer;

/**
 * 生产者
 */

public class Producer {

    /**
     * 销售
     * @param money
     */
    public void sellProduct(float money){
        System.out.println("销售产品,并拿到钱:"+money);
    }

    /**
     * 售后
     * @param money
     */
    public void afterService(float money){
        System.out.println("提供售后服务,并拿到钱:"+money);
    }


}

客户通过代理商购买物品

package com.czy.cglib;

import com.czy.proxy.IProducer;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

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

/**
 * 模拟一个消费者
 */
public class Client {
    public static void main(String[] args) {
        final Producer producer = new Producer();
        Producer cglibProducer = (Producer) Enhancer.create(producer.getClass(), new MethodInterceptor() {
            /**
             * 执行对象的任何方法都会经过该方法
             * @param o 同invoke的proxy,代理对象的引用
             * @param method 执行的方法
             * @param objects 方法的参数
             * @param methodProxy 当前执行方法的代理对象
             * @return
             * @throws Throwable
             */
            public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
                //提供增强代码
                Float money = (Float)objects[0];
                if("sellProduct".equals(method.getName())){
                    //代理商吃了二成回扣
                    return method.invoke(producer,money*0.8f);
                }
                return method.invoke(producer,objects);
            }
        });
        cglibProducer.sellProduct(10000f);
    }
}

posted @   czyaaa  阅读(51)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· Docker 太简单,K8s 太复杂?w7panel 让容器管理更轻松!
点击右上角即可分享
微信分享提示