设计模式之代理模式

一、静态代理
这种模式的关键是真正的目标对象和代理对象得实现同一个接口,同时代理对象中要注入一个目标对象得引用,调用代理对象之后,代理对象内部调用目标对象。相关代码如下:
1、定义一个接口 代理对象和目标对象(被代理对象)都要实现这同一个接口

interface Service{
    void m1();
    void m2();
    void m3();
}

2、被代理对象实现Service接口

//目标对象  被代理对象
class Target implements Service{

    @Override
    public void m1() {
        System.out.println("This is Target m1()");
    }

    @Override
    public void m2() {
        System.out.println("This is Target m2()");
    }

    @Override
    public void m3() {
        System.out.println("This is Target m3()");
    }
}

3、代理对象同样也需要实现公共接口

//代理对象
class StaticProxy implements Service{

    Service target;  //注入目标对象

    public StaticProxy(Service target) {
        this.target = target;
    }

    @Override
    public void m1() {
        System.out.println("静态代理--开始");
        target.m1();
        System.out.println("静态代理--结束");
    }

    @Override
    public void m2() {
        System.out.println("静态代理--开始");
        target.m2();
        System.out.println("静态代理--结束");
    }

    @Override
    public void m3() {
        System.out.println("静态代理--开始");
        target.m3();
        System.out.println("静态代理--结束");
    }
}

4、测试用例

public class A_Test003_StaticProxy {

    public static void main(String[] args) {
        Service target = new Target();
        Service staticProxy = new StaticProxy(target);
        staticProxy.m1();
    }
}

5、运行结果
在这里插入图片描述

二、动态代理
动态代理可分为JDK动态代理和CGLIB动态代理两种,JDK动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理。而CGLIB动态代理是将代理对象类的class文件加载进来,通过修改其字节码生成子类来处理。
1、如果目标对象实现了接口,默认情况下会采用JDK的动态代理实现AOP
2、如果目标对象实现了接口,可以强制使用CGLIB实现AOP
3、如果目标对象没有实现了接口,必须采用CGLIB库,spring会自动在JDK动态代理和CGLIB之间转换

JDK动态代理和CGLIB动态代理的区别?
(1)JDK动态代理只能对实现了接口的类生成代理,而不能针对类
(2)CGLIB是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法
因为是继承,所以该类或方法最好不要声明成final
1、JDK动态代理
1、定义接口

interface Service2{
    void m1();
    void m2();
    void m3();
}

2、被代理对象实现公共接口

//目标对象  被代理对象
class Target2 implements Service2{

    @Override
    public void m1() {
        System.out.println("This is Target2 m1()");
    }

    @Override
    public void m2() {
        System.out.println("This is Target2 m2()");
    }

    @Override
    public void m3() {
        System.out.println("This is Target2 m3()");
    }
}

3、代理对象实现InvocationHandler接口

//代理对象实现InvocationHandler接口
class JdkProxy implements InvocationHandler {

    private Object target ;//目标对象  被代理对象

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("jdk动态代理--开始");
        Object result = method.invoke(target, args);
        System.out.println("jdk动态代理--结束");
        return result;
    }
    //定义获取代理对象方法
    public Object getJDKProxy(Object target){
        //为目标对象target赋值
        this.target = target;
        //JDK动态代理只能针对实现了接口的类进行代理,newProxyInstance 函数所需参数就可看出
        return Proxy.newProxyInstance(
                target.getClass().getClassLoader(),
                target.getClass().getInterfaces(),
                this); //this表示当前代理对象即为JdkProxy
    }
}

4、运行结果
在这里插入图片描述

2、CGLIB动态代理
1、定义接口

interface Service3{
    void m1();
    void m2();
    void m3();
}

2、被代理对象

//目标对象  被代理对象
class Target3 implements Service3{

    @Override
    public void m1() {
        System.out.println("This is Target3 m1()");
    }

    @Override
    public void m2() {
        System.out.println("This is Target3 m2()");
    }

    @Override
    public void m3() {
        System.out.println("This is Target3 m3()");
    }
}

4、代理对象


//Cglib动态代理,实现MethodInterceptor接口
class CglibProxy implements MethodInterceptor {

    private Object target;//需要代理的目标对象

    //重写拦截方法
    @Override
    public Object intercept(Object obj, Method method, Object[] arr, MethodProxy proxy) throws Throwable {
        System.out.println("Cglib动态代理--开始");
        Object invoke = method.invoke(target, arr);//方法执行,参数:target 目标对象 arr参数数组
        System.out.println("Cglib动态代理--结束");
        return invoke;
    }
    //定义获取代理对象方法
    public Object getCglibProxy(Object objectTarget){
        //为目标对象target赋值
        this.target = objectTarget;
        Enhancer enhancer = new Enhancer();
        //设置父类,因为Cglib是针对指定的类生成一个子类,所以需要指定父类
        enhancer.setSuperclass(objectTarget.getClass());
        enhancer.setCallback(this);// 设置回调
        Object result = enhancer.create();//创建并返回代理对象
        return result;
    }

}

4、运行结果
在这里插入图片描述

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值