设计模式之代理模式

目录

1、静态代理(装饰器模式)

2、jdk动态代理

3、cjlib动态代理


概述:代理模式分为静态代理动态代理,其中动态代理又包括jdk动态代理cjlib动态代理,Spring框架底层用到了很多动态代理。

本质:使用代理对象增强被代理对象的方法。增强的意思就是在原有方法前后执行一些其他自定义的方法。

案例:Spring AOP、拦截器

1、静态代理(装饰器模式)

既然叫代理,代理对象与被代理对象具有相同的行为功能。所以一般情况下,静态代理中代理对象和被代理对象都属同一接口。之所以叫静态代理,不是“随类的加载而加载”的意思,而是“固定的”,是因为一类对象都对应一个代理对象,这种创建代理对象的方法太过麻烦,总要创建各种类,可不可以仅写一个类就可以创建任意代理对象?使用动态代理实现。下面先列举静态代理demo

 接口定义行为功能

package com.demo.statics;

public interface Man {
    void speak();
}

 被代理类

package com.demo.statics;

/**
 * 被代理类
 */
public class BaoQiang implements Man {
    @Override
    public void speak() {
        System.out.println("我是宝强");
    }
}

 代理类

package com.demo.statics;

/**
 * 代理类
 */
public class Songjie implements Man {

    private Man man;

    public Songjie(Man man) {
        this.man = man;
    }

    @Override
    public void speak() {
        System.out.println("我是宋jie代理");
        man.speak();
    }
}

 测试

package com.demo.statics;

public class MainTest {

    public static void main(String[] args) {
        Songjie proxy = new Songjie(new BaoQiang());
        proxy.speak();
    }
}

 运行结果

2、jdk动态代理

 根据jdk提供的api创建代理对象。

动态代理对象的类型与常规类不一样,类似下面这种

 jdk动态代理的被代理类必须要实现接口,$Proxy0能与实现接口相互转换,如果没实现接口会类型转异常。如果被代理类实现多个接口,那么$Proxy0可以与任意接口相互转换。

demo

接口定义行为功能

package com.demo.dynamic;

public interface God {
    String fly();
}
package com.demo.dynamic;

public interface Man {
    String speak();
}

 被代理类

package com.demo.dynamic;

/**
 * 被代理类
 */
public class BaoQiang implements Man, God {
    @Override
    public String speak() {
        return "我是宝强";
    }

    @Override
    public String fly() {
        return "我会飞";
    }
}

 使用jdk底层api创建代理对象

package com.demo.dynamic;

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

public class JdkProxy implements InvocationHandler {

    private Object target;

    public JdkProxy(Object target) {
        this.target = target;
    }

    public static <T> T createProxy(T t) {
        /**
         * ClassLoader loader 被代理类的类加载器
         * Class<?>[] interfaces 被代理类实现的所有接口
         * InvocationHandler h 执行处理器,调用目标方法时,会被InvocationHandler拦截
         */
        Object o = Proxy.newProxyInstance(t.getClass().getClassLoader(), t.getClass().getInterfaces(), new JdkProxy(t));
        return (T) o;
    }

    /**
     * 代理对象执行目标方法时都会被InvocationHandler拦截,执行invoke方法
     *
     * @param proxy
     * @param method
     * @param args
     * @return
     * @throws Throwable
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("增强方法---");
        //执行真正的被代理对象的方法
        Object invoke = method.invoke(target, args);
        return "返回值:" + invoke;
    }
}

测试类

package com.demo.dynamic;

public class MainTest {
    public static void main(String[] args) {
        Man proxy = JdkProxy.createProxy(new BaoQiang());
        System.out.println(proxy.speak());
        System.out.println(((God) proxy).fly());
    }
}

 运行结果

3、cjlib动态代理

上述jdk动态代理的实现必须给被代理类实现接口,如果我不想实现接口也具有代理功能,就可以使用cjlib动态代理。它是根据cjlib包中api创建代理对象的,要先引入包然后使用增求器Enhancer来创建代理对象。

demo

被代理类

package com.demo.cjlib;

/**
 * 被代理类
 */
public class BaoQiang {
    public String speak() {
        return "我是宝强";
    }
}

 使用cjlib底层api创建代理对象

package com.demo.cjlib;

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

import java.lang.reflect.Method;

public class CjlibProxy {

    public static <T> T createProxy(T t) {
        // 1、创建增强器
        Enhancer enhancer = new Enhancer();
        // 2、设置父类,使代理类具有父类的方法
        enhancer.setSuperclass(t.getClass());
        // 3、设置回调,当代理对象调用方法时,会被回调拦截
        enhancer.setCallback(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 {
                System.out.println("cjlib增强方法---");
                Object o1 = methodProxy.invokeSuper(o, objects);
                return "返回值:" + o1;
            }
        });

        Object o = enhancer.create();
        return (T) o;
    }
}

 测试类

package com.demo.cjlib;

public class MainTest {
    public static void main(String[] args) {
        BaoQiang proxy = CjlibProxy.createProxy(new BaoQiang());
        System.out.println(proxy.speak());
    }
}

运行结果

代理模式是一种结构型设计模式,它提供一个代理对象来代表另一个对象。在代理模式中,有一个被称为实际对象(Subject)和一个被称为代理对象(Proxy)的中介,代理对象持有实际对象的引用,并且可以控制对实际对象的访问。代理模式的主要目的是在不修改原始对象的情况下,为原始对象添加额外的逻辑处理。 代理模式分为多种类型,如远程代理、虚拟代理、保护代理等,它们各自有不同的应用场景: - 远程代理:为远程对象提供一个本地代表。 - 虚拟代理:根据需要创建开销大的对象,通过虚拟代理控制访问这些对象的过程。 - 保护代理:控制对原始对象的访问权限,例如进行权限检查。 代理模式的优点包括: 1. 能够控制对真实对象的访问,并在访问前后添加额外的逻辑。 2. 可以通过代理对象实现延迟加载,即在实际需要时才创建真实对象。 3. 增强了对真实对象的封装,并且可以避免对真实对象的重复引用。 在C#中实现代理模式通常涉及以下步骤: 1. 定义一个接口或抽象类,声明真实对象和代理对象需要实现的方法。 2. 实现真实对象的类,按照接口或抽象类的要求实现具体方法。 3. 实现代理类,它同样实现接口或抽象类,并在方法中持有真实对象的引用,通过调用真实对象的方法来执行所需的操作,同时可以添加额外的逻辑。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值