代理模式与动态代理

什么是代理模式?

代理,在生活中很常见,比如我不想去做一些复杂的,多重选择的事情,或者以不同的方式回应不同的事件,这个时候我们就可以找一个代理,让他帮我做这些事,只需要给他需要的东西,他就能自动匹配我想让他做的事情,不需要我亲自去不同的地方做不同的事情。

代理分为静态代理和动态代理

静态代理

静态代理,在编译期间就确定了代理关系的代理类叫静态代理。即需要手动实现代理,为每个业务类抽象一个接口,对应的创建代理类

写一个传菜的服务员

//定义一个厨师接口
public interface Cook {
    String send(String food);
}

//厨师出菜
public class CookImpl implements cook {
    public String send(String food) {
        System.out.println("send food:" + food);
        return food;
    }
}

//服务员(代理)代替厨师传菜
public class Waiter implements Cook {

    private final Cook cook;

    public CookProxy(Cook cook) {
        this.cook = cook;
    }

    //增强了厨师出菜的方法
    @Override
    public String send(String food) {
        //调用方法之前,我们可以添加自己的操作
        System.out.println("接收到麻辣香锅");
        smsService.send(food);
        //调用方法之后,我们同样可以添加自己的操作
        System.out.println("将麻辣香锅传给客人");
        return null;
    }
}

我们每多出一个厨子,就要多出一个服务员传菜,厨子和服务员一对一,很不实用,我们想要很多厨师,但只需要一个服务员传菜,这时候就需要动态代理出马了。

静态代理的缺点:

  • 1、重复性:即使有多个基础业务需要代理,也不需要编写过多重复的模板代码;

  • 2、脆弱性:当基础接口变更时,同步改动代理并不是必须的。

动态代理

动态代理是指在运行时才确定代理类和被代理类的关系,即我的服务员要对接哪个厨子,这个是由用户选择的,用户传入什么参数,调用什么方法,返回值类型,来确定到底需要用到哪些方法。其实现有JDK动态代理、Javassist 动态代理和 ASM 动态代理等。

在动态代理中最核心的方法就是Proxy类中的newProxyInstance方法InvocationHandler 方法

就拿JDK动态代理举例:

public static Object newProxyInstance(ClassLoader loader,//类加载器,用于加载代理对象。
                                          Class<?>[] interfaces,//被代理类实现的一些接口
                                          InvocationHandler h)//实现了 InvocationHandler 接口的对象
        throws IllegalArgumentException
    {
        ......
    }

要实现动态代理就要实现InvocationHandler中的invoke方法

public interface InvocationHandler {

    /**
     * 当你使用代理对象调用方法的时候实际会调用到这个方法
     */
    public Object invoke(Object proxy, Method method, Object[] args)
        throws Throwable;
}

invoke对象的三个参数是,proxy动态生成的代理类,method代理类调用的方法,args是method方法的参数

通过Proxy 类的 newProxyInstance() 创建的代理对象在调用方法的时候,实际会调用到实现InvocationHandler 接口的类的 invoke()方法。 你可以在 invoke() 方法中自定义处理逻辑,比如在方法执行前后做什么事情,对实际想调用的方法进行增强。

实例


 HWSale saleProxy = (HWSale) Proxy.newProxyInstance(
                hwSale.getClass().getClassLoader(), // 通过真实对象获取对应的类加载器
                hwSale.getClass().getInterfaces(),  // 通过真实对象获取对应的接口 Class 数组
                   //匿名内部类实现了InvocationHandler方法,对方法的增强
                 new InvocationHandler() {
                    @Override        //这里可以实现方法的动态选择,即在运行时才确定代理对象
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        String s1 = null;
                        // 发现当前执行的方法是 sale 方法
                        if (method.getName().equals("salePhone")) {
                            // 获取当前方法执行的实际参数,关注方法的参数真实数据类型和个数
                            Float proxyPrice = (Float) args[0];

                            // 真实对象依然执行目标方法,但是方法的参数是修改之后的代理价格【增强参数】
                            // 得到了原本的返回值类型
                            s1 = hwSale.salePhone(proxyPrice * 0.9F);

                            // 【增强返回值】
                            s1 += "华为手机打九折啦";

                            return s1;
                        } else {
                            return null;
                        }
                    }
                });

        System.out.println("代理售出");
        String s1 = saleProxy.salePhone(10000F);
        System.out.println(s1);
    }

输出结果为

9000.00华为手机打九折啦
//或者
代理售出
10000.0

输出结果取决于方法名是否为salePhone

总结

静态代理在设计模式中随处可见,但静态代理有重复性和脆弱性 动态代理是用来给方法实现增强并且让方法选择更为多样的,即在用户调用一个代理类中的方法就能够执行相对应的方法,甚至通过返回值类型来确定执行哪一个重载的方法。

如有不足,私信或者评论指正,肥肠感激。🙋

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值