代理模式

一、简介

定义:由于某些原因需要给某对象提供一个代理以控制对该对象的访问。这时,访问对象不适合或者不能直接引用目标对象,代理对象作为访问对象和目标对象之间的中介。
场景:远程代理,这种方式通常是为了隐藏目标对象存在于不同地址空间的事实,方便客户端访问。虚拟代理,这种方式通常用于要创建的目标对象开销很大时。安全代理,这种方式通常用于控制不同种类客户对真实对象的访问权限。智能指引,主要用于调用目标对象时,代理附加一些额外的处理功能。延迟加载,指为了提高系统的性能,延迟对目标的加载。

  • 角色结构:
  • 抽象主题(Subject)类:通过接口或抽象类声明真实主题和代理对象实现的业务方法。
  • 真实主题(Real Subject)类:实现了抽象主题中的具体业务,是代理对象所代表的真实对象,是最终要引用的对象。
  • 代理(Proxy)类:提供了与真实主题相同的接口,其内部含有对真实主题的引用,它可以访问、控制或扩展真实主题的功能。

二、模式实现

以车票平台代理车票销售为例子

public interface Ticket { -> 抽象主题
    String name();
    Double price();
    void sale();
}

public class BusTicket implements Ticket { -> 真实主题
    @Override
    public String name() {
        return "汽车票";
    }

    @Override
    public Double price() {
        return 10.00;
    }

    @Override
    public void sale() {
        System.out.println(name() + ":" + price());
    }
}

public class TicketPlatform implements Ticket { -> 代理
    private Ticket ticket;

    public TicketPlatform(Ticket ticket) {
        this.ticket = ticket;
    }

    @Override
    public String name() {
        return ticket.name();
    }

    @Override
    public Double price() {
        return ticket.price();
    }

    public void sale() {
        ticket.sale();
    }
}

Ticket ticket = new BusTicket(); -> 静态代理
TicketPlatform platform = new TicketPlatform(ticket);
platform.sale();

三、动态代理

JDK代理实现

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

public class DynamicPlatform implements InvocationHandler {
    private Object object; -> 被代理对象

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

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("jdk代理前");
        Object obj = method.invoke(object, args);
        System.out.println("jdk代理后");
        return obj;
    }
}

Ticket ticket = new BusTicket();
DynamicPlatform platform = new DynamicPlatform(ticket);
ClassLoader loader = ticket.getClass().getClassLoader();
Ticket proxy = (Ticket) Proxy.newProxyInstance(loader,
        new Class[]{Ticket.class}, platform);
proxy.sale();

Cglib代理实现

 -> pom.xml
    cglib
    cglib
    3.3.0

import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;

public class CglibDynamicProxy implements MethodInterceptor {
    private Object object; -> 被代理对象 

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

    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        System.out.println("cglib代理前");
        Object proxy = methodProxy.invoke(object, objects);
        System.out.println("cglib代理后");
        return proxy;
    }
}

Ticket ticket = new BusTicket();
CglibDynamicProxy cglibDynamicProxy = new CglibDynamicProxy(ticket);
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(ticket.getClass());
enhancer.setCallback(cglibDynamicProxy);
Ticket proxy = (Ticket) enhancer.create();
proxy.sale();

你可能感兴趣的:(代理模式)