23种设计模式之代理模式

代理模式

简介

代理模式,就是给一个对象提供一个代理对象,并由代理对象控制对原对象的引用

代理模式流程图

分类

1. 静态代理

1.1静态代理角色分析

  • 客户 : 使用代理角色来进行一些操作
  • 代理角色 : 代理真实角色 ; 代理真实角色后 , 一般会做一些附属的操作 。
  • 真实角色 : 被代理的角色
  • 抽象角色 : 一般使用接口或者抽象类来实现

1.2 代码实现

1.Rent.java :抽象角色

//抽象角色:租房
public interface Rent {
    public void rent();
}

2.LandLord.java: 真实角色

//真实角色: 房东 出租房子
public class LandLord implements Rent{
    @Override
    public void rent() {
        System.out.println("出租房屋");
    }
}

3.Proxy.java :代理对象

//代理角色:房屋中介
public class Proxy implements Rent {

    private  LandLord landLord;

    public Proxy() {
    }

    public Proxy(LandLord landLord) {
        this.landLord = landLord;
    }

    //租房
    @Override
    public void rent() {
        landLord.rent();
        seeHouse();
        fee();
    }

    //看房
    public void seeHouse(){
        System.out.println("带房客看房");
    }

    //收中介费
    public void fee(){
        System.out.println("收中介费");
    }
}

4.Client.java :客户

//客户类,一般客户都会去找代理!
public class Client {
    public static void main(String[] args) {
        //房东要租房
        LandLord landLord = new LandLord();
        //中介帮助房东
        Proxy proxy = new Proxy(landLord);
        //客户找中介租房
        proxy.rent();
    }
}

分析代码:
在这个过程中,你直接接触的就是中介。你看不到房东,但是你通过中介代理依旧租到了房东的房子,这就是所谓的代理模式。

1.3 优点:

  • 可以做到在符合开闭原则的情况下对目标对象进行功能扩展
  • 公共的业务由代理来完成,实现了业务的分工
  • 公共业务发生扩展时变得更加集中和方便

1.4 缺点 :

  • 多了代理类 , 工作量变大了 , 开发效率降低

2.动态代理

我们想要静态代理的好处,又不想要静态代理的缺点,所以 , 就有了动态代理 !

2.1 简介

动态代理的角色和静态代理的一样。

动态代理的代理类是动态生成的 , 静态代理的代理类是我们提前写好的。

动态代理分为两类 : 一类是基于接口动态代理 , 一类是基于类的动态代理

  • 基于接口的动态代理----JDK动态代理
  • 基于类的动态代理–cglib

核心 : InvocationHandler【调用处理程序】和Proxy【代理】

2.2 代码实现

抽象角色和真实角色和之前的一样。
1.Rent.java :抽象角色

//抽象角色:租房
public interface Rent {
    public void rent();
}

2.LandLord.java: 真实角色

//真实角色: 房东 出租房子
public class LandLord implements Rent{
    @Override
    public void rent() {
        System.out.println("出租房屋");
    }
}

3.ProxyInvocationHandler. java 即代理角色

//代理角色
public class ProxyInvocationHandler implements InvocationHandler {
    private Rent rent;

    public void setRent(Rent rent) {
        this.rent = rent;
    }

    //生成代理类,重点是第二个参数,获取要代理的抽象角色!
    // 之前都是一个角色,现在可以代理一类角色
    public Object getProxy(){
        return Proxy.newProxyInstance(this.getClass().getClassLoader(),rent.getClass().getInterfaces(),this);
    }

    // proxy : 代理类 method : 代理类的调用处理程序的方法对象.
    // 处理代理实例上的方法调用并返回结果
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        seeHouse();
        fee();
        //核心:本质利用反射实现!
        Object invoke = method.invoke(rent, args);
        return invoke;
    }
    //看房
    public void seeHouse(){
        System.out.println("带房客看房");
    }

    //收中介费
    public void fee(){
        System.out.println("收中介费");
    }
}

4.Client . java 用户

//租客
public class Client {
    public static void main(String[] args) {
        //真实角色
        LandLord landLord =new LandLord();
        //代理实例的调用处理程序
        ProxyInvocationHandler pih = new ProxyInvocationHandler();
        pih.setRent(landLord); //将真实角色放置进去!
        Rent proxy = (Rent)pih.getProxy(); //动态生成对应的代理类!
        proxy.rent();
    }
}

核心:一个动态代理 , 一般代理某一类业务 。一个动态代理可以代理多个类,代理的是接口!

注意
Proxy.newProxyInstance()方法接受三个参数:

ClassLoader loader:指定当前目标对象使用的类加载器,获取加载器的方法是固定的
Class<?>[] interfaces:指定目标对象实现的接口的类型,使用泛型方式确认类型
InvocationHandler:指定动态处理器,执行目标对象的方法时,会触发事件处理器的方法

2.3 优点

静态代理有的它都有,静态代理没有的,它也有

  • 可以做到在符合开闭原则的情况下对目标对象进行功能扩展
  • 公共的业务由代理来完成,实现了业务的分工
  • 公共业务发生扩展时变得更加集中和方便
  • 一个动态代理 , 一般代理某一类业务。
  • 一个动态代理可以代理多个类,代理的是接口!

3. Cglib代理

3.1CgLib其原理:

通过字节码技术为一个类创建子类,并在子类中采用方法****拦截的技术拦截所有父类方法的调用。但因为采用的是继承,所以不能对final修饰的类进行代理。

3.2 代码

1.创建CgLib代理类

public class CglibProxy implements MethodInterceptor {
    private Object target;
    public Object getInstance(final Object target) {
        this.target = target;
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(this.target.getClass());
        enhancer.setCallback(this);
        return enhancer.create();
    }

    public Object intercept(Object object, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
        System.out.println("买房前准备");
        Object result = methodProxy.invoke(object, args);
        System.out.println("买房后装修");
        return result;
    }
}

2.测试

public class CglibProxyTest {
    public static void main(String[] args){
        BuyHouse buyHouse = new BuyHouseImpl();
        CglibProxy cglibProxy = new CglibProxy();
        BuyHouseImpl buyHouseCglibProxy = (BuyHouseImpl) cglibProxy.getInstance(buyHouse);
        buyHouseCglibProxy.buyHosue();
    }
}

3.3 总结:

  • CgLib创建的动态代理对象比JDK创建的动态代理对象的性能更高

  • 但是CGLIB创建代理对象时所花费的时间却比JDK更

  • 对于单例的对象,因为无需频繁创建对象,用CgLib合适,反之使用JDK方式要更为合适一些

  • 同时由于CgLib是采用动态创建子类的方法,对于final修饰的方法无法进行代理。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值