设计模式--第11篇(代理模式)

一,代理模式

代理模式:

  1. 为对象提供一个替身,以控制对这个对象的访问。即通过代理对象访问目标对象,可以在目标对象的基础上实现扩展,增强额外的功能;
  2. 被代理对象可以是远程对象、创建开销大的对象或需要安全控制的对象等;
  3. 主要有:静态代理,jdk动态代理,Cglib动态代理

二,原理类图

在这里插入图片描述
意图: 为其他对象提供一种代理以控制对这个对象的访问。
适用性:
在需要用比较通用和复杂的对象指针代替简单的指针的时候,使用Proxy模式。下面是一 些可以使用Proxy 模式常见情况:

  1. 远程代理(Remote Proxy )为一个对象在不同的地址空间提供局部代表。 NEXTSTEP[Add94] 使用NXProxy 类实现了这一目的。Coplien[Cop92] 称这种代理为“大使” (Ambassador )。
  2. 虚代理(Virtual Proxy )根据需要创建开销很大的对象。在动机一节描述的ImageProxy 就是这样一种代理的例子。
  3. 保护代理(Protection Proxy )控制对原始对象的访问。保护代理用于对象应该有不同 的访问权限的时候。例如,在Choices 操作系统[ CIRM93]中KemelProxies为操作系统对象提供 了访问保护。
  4. 智能指引(Smart Reference )取代了简单的指针,它在访问对象时执行一些附加操作。 它的典型用途包括:对指向实际对象的引用计数,这样当该对象没有引用时,可以自动释放它(也称为SmartPointers[Ede92 ] )。
    当第一次引用一个持久对象时,将它装入内存。
    在访问一个实际对象前,检查是否已经锁定了它,以确保其他对象不能改变它。

三,实例

静态代理

静态代理: 定义接口或者父类,被代理对象和代理对象一起实现或继承;
优点: 在不修改被代理类的前提下实现功能扩展增强;
缺点: 同时需要继承或实现同一个接口,一旦接口方法增加所有代理和被代理类都需要调整;

接口

package com.neei.proxy.statis;

/**
 * @param
 * @Author: AaNeei
 * @Date: 2019/10/11  21:35
 * @Description: 游学网
 * @throws:
 */
public interface ITeacher {
    void teach();
}

被代理类

package com.neei.proxy.statis;

/**
 * @param
 * @Author: AaNeei
 * @Date: 2019/10/11  21:34
 * @Description: 游学网
 * @throws:
 */
public class Teacher implements ITeacher {
    @Override
    public void teach(){
        System.out.println("老师上课中。。。。");
    }
}

代理类

package com.neei.proxy.statis;

/**
 * @param
 * @Author: AaNeei
 * @Date: 2019/10/11  21:36
 * @Description: 游学网
 * @throws:
 */
public class TeacherProxy implements ITeacher {
    private ITeacher iTeacher;

    public TeacherProxy(ITeacher iTeacher) {
        this.iTeacher = iTeacher;
    }

    @Override
    public void teach() {
        System.out.println("静态代理开始-----功能增强");
        iTeacher.teach();
        System.out.println("功能增强完毕-----静态代理结束");

    }
}

调用

package com.neei.proxy.statis;

/**
 * @param
 * @Author: AaNeei
 * @Date: 2019/10/11  21:38
 * @Description: 游学网
 * @throws:
 */
public class Client {
    public static void main(String[] args) {
        Teacher teacher = new Teacher();
        TeacherProxy proxy=new TeacherProxy(teacher);
        proxy.teach();
    }
}
JDK动态代理

JDK动态代理:

  1. 代理对象不需要实现接口,被代理对象需要实现接口,又叫接口代理
  2. 代理对象的生成是通过JDK的API,动态的在内存中构建的;

接口

package com.neei.proxy.statis;

/**
 * @param
 * @Author: AaNeei
 * @Date: 2019/10/11  21:35
 * @Description: 游学网
 * @throws:
 */
public interface ITeacher {
    void teach();
}

被代理类

package com.neei.proxy.statis;

/**
 * @param
 * @Author: AaNeei
 * @Date: 2019/10/11  21:34
 * @Description: 游学网
 * @throws:
 */
public class Teacher implements ITeacher {
    @Override
    public void teach(){
        System.out.println("老师上课中。。。。");
    }
}

代理类

package com.neei.proxy.dynamic;

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

/**
 * @param
 * @Author: AaNeei
 * @Date: 2019/10/11  21:56
 * @Description: 游学网
 * @throws:
 */
public class ProxyFactory {

    private Object object;

    public ProxyFactory(Object object) {
        this.object = object;
    }

    public Object getProxyInstance() {
        return Proxy.newProxyInstance(
        		//代理对象的类加载器
                object.getClass().getClassLoader(),
                //代理对象实现的接口集合
                object.getClass().getInterfaces(),
                //方法处理拦截器
                new InvocationHandler() {
                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        System.out.println("JDK动态代理开始---");
                        return method.invoke(object, args);
                    }
                });
    }
}


调用

package com.neei.proxy.dynamic;

/**
 * @param
 * @Author: AaNeei
 * @Date: 2019/10/11  21:55
 * @Description: 游学网
 * @throws:
 */
public class Client {
    public static void main(String[] args) {
        ITeacher teacher=new Teacher();
        ITeacher instance = (ITeacher) new ProxyFactory(teacher).getProxyInstance();
        instance.teach();
    }
}

CGLIB动态代理

CGLIB动态代理:

  1. 静态代理和jdk动态代理都需要实现接口,如果没有接口实现的时候就只能使用cglib代理;
  2. cglib又叫子类代理,在内存中构建一个子类对象(不能是final类),实现对目标对象的功能扩展;
  3. cglib是个高性能的代码生成包,可以在运行期扩展java类实现、java接口等,多用于AOP框架,实现方法拦截;
  4. cglib底层采用了asm框架的字节码技术转换字节码并生成新的类;

被代理类

package com.neei.proxy.statis;

/**
 * @param
 * @Author: AaNeei
 * @Date: 2019/10/11  21:34
 * @Description: 游学网
 * @throws:
 */
public class Teacher implements ITeacher {
    @Override
    public void teach(){
        System.out.println("老师上课中。。。。");
    }
}

代理类

package com.neei.proxy.dynamic;

import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;

import java.lang.reflect.Method;

/**
 * @param
 * @Author: AaNeei
 * @Date: 2019/10/11  21:56
 * @Description: 游学网
 * @throws:
 */
public class ProxyFactory implements MethodInterceptor {

    private Object object;

    public ProxyFactory(Object object) {
        this.object = object;
    }

    public Object getProxyInstance() {
        //创建工具类
        Enhancer enhancer = new Enhancer();
        //设置目标类
        enhancer.setSuperclass(object.getClass());
        //回调目标类方法
        enhancer.setCallback(this);
        //返回代理对象
        return enhancer.create();

    }

    @Override
    public Object intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {

        System.out.println("cglib动态代理开始----");
        Object invoke = method.invoke(object, args);
        return invoke;
    }
}

调用

package com.neei.proxy.dynamic;

/**
 * @param
 * @Author: AaNeei
 * @Date: 2019/10/11  21:55
 * @Description: 游学网
 * @throws:
 */
public class Client {
    public static void main(String[] args) {

        Teacher teacher=new Teacher();
        Teacher instance = (Teacher) new ProxyFactory(teacher).getProxyInstance();
        instance.teach();
    }
}

四,源码分析

JDK源码中使用的代理模式,如Proxy.newProxyInstance

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章