设计模式-代理模式

代理模式

应用场景:不该变源代码,给源代码增加功能。

在Java中,常用的代理有JDK动态代理和cglib动态代理。JDK动态代理由java.lang.reflect包下的Proxy提供的,基于接口生成的代理类,实现对目标类的代理;cglib是由Oracle提供的Java开发工具包,采用字节码技术,直接修改字节码并生成代理子类。

注意:JDK动态代理中,调用方法时其代理类并不知道目标类的具体方法,因此需要通过反射来进行调用,效率很低。

JDK实现动态代理

java 复制代码
public class Person implements IPerson {
    @Override
    public void eat(){
        System.out.println("开始吃饭");
    }
}

1.定义接口,定义需要代理的方法

java 复制代码
public interface IPerson {
    public void eat();
}

2.定义类并创建代理对象

java 复制代码
    public static IPerson createPersonProxy(Person person){
        IPerson iPersonProxy = (IPerson) Proxy.newProxyInstance(PersonProxy.class.getClassLoader()
                , new Class[]{IPerson.class}
                , new InvocationHandler() {
                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        if(method.getName() == "eat"){
                            System.out.println("拿筷子");
                            System.out.println("拿碗");
                        }

                        return method.invoke(person ,args);
                    }
                });
        return iPersonProxy;
    }
}

3.调用代理对象的方法

java 复制代码
public class Demo {
    public static void main(String[] args) {
        Person person = new Person();
        IPerson personProxy = PersonProxy.createPersonProxy(person);
        personProxy.eat();
    }
}

CGLIB实现动态代理

java 复制代码
public class Person{
    public void eat(){
        System.out.println("开始吃饭");
    }
}
java 复制代码
public class PersonProxy implements MethodInterceptor {
    private Enhancer enhancer = new Enhancer();

    public Object getProxy(Class clazz){
        enhancer.setSuperclass(clazz);
        enhancer.setCallback(this);
        //通过字节码技术动态创建子类实例
        return enhancer.create();
    }

    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        if(method.getName().equals("eat")){
            System.out.println("拿筷子");
            System.out.println("拿碗");
        }

        //通过代理类调用父类中的方法
        Object result = methodProxy.invokeSuper(o, objects);

        return result;
    }
}
相关推荐
绿豆人1 天前
Go设计模式学习
学习·设计模式·golang
逮到647了1 天前
23种设计模式简述
设计模式
爱吃烤鸡翅的酸菜鱼1 天前
【Java】封装位运算通用工具类——用一个整数字段替代几十个布尔列,极致节省存储空间
java·开发语言·设计模式·工具类·位运算·合成复用原则
geovindu1 天前
go: Model,Interface,DAL ,Factory,BLL using mysql
开发语言·mysql·设计模式·golang·软件构建
guojb8241 天前
当 Vue 3 遇上桥接模式:手把手教你优雅剥离虚拟滚动的业务大泥球
vue.js·设计模式
我登哥MVP1 天前
【Spring6笔记】 - 15 - Spring中的八大设计模式
java·spring boot·笔记·spring·设计模式·intellij-idea
无籽西瓜a1 天前
【西瓜带你学设计模式 | 第十六期 - 迭代器模式】迭代器模式 —— 统一遍历实现、优缺点与适用场景
java·后端·设计模式·迭代器模式·软件工程
程序员小寒1 天前
JavaScript设计模式(十):模板方法模式实现与应用
前端·javascript·设计模式·模板方法模式
likerhood1 天前
关于三种工厂的设计模式总结
设计模式
榴莲omega1 天前
第14天:React 工程化与设计模式
前端·react.js·设计模式