3.代理模式——Proxy

介绍

代理模式在java中分动态代理和静态代理两种,是一种较为常见的设计模式,顾名思义,它主要的形式是通过一个代理的方式去访问和操作被代理的对象,如下UML图。那么为什么使用这种模式呢,这样既增加了接口和代理类,又使得调用的链拉的更长,仿佛增加了逻辑的复杂度,直接操作对象岂不是更好?


image.png

优点

  1. 职责清晰:真实的角色就是实现业务逻辑,不关心其他非本职责的事务。隐藏具体实现细节,是迪米特法则的一种体现;
  2. 高扩展性:具体实现随时都会发生变化,只要接口在,代理类就可以在不做任何修改的情况下使用。我觉得这也正好顺应了开闭原则。

动态代理和静态代理

静态代理:代理者的代码是程序员自己或通过自动生成工具的代码再对其进行编译。在代码运行之前,代理类的class编译文件已经存在。
动态代理:通过反射机制动态生成代理者对象,在code阶段不需要知道代理谁,将在执行阶段决定。通过提供的InvocationHandler接口的invoke方法来决定被代理的真实方法。使我们的代码逻辑更简单。

静态代理的缺点只能为给定的实现类做代理,如果接口不同那么需要重新定义不同代理类,较为复杂,但静态代理更符合面向对象原则。

使用场景

  1. 想要隐藏某个类时,可以对这个类提供代理类;
  2. 一个类对不同的调用者提供不同的调用权限或方式;
  3. 代理过程中添加额外操作。

应用实例

  • AndoridManagerService;(静态代理)
  • Retrofit对OkHttp3的封装;(动态代理)
  • 一些插件化框架对ActivityManager的代理和Intent的替换。Hook技术实现Android插件化

动态代理的实现

  1. 定义接口
public interface IGame {
    void play();
    String getLevel(int exp);
}
  1. 实现类
public class GameImpl implements IGame {
    public void play() {
        System.out.println("开始游戏!");
    }

    @Override
    public String getLevel(int exp) {
        return String.valueOf(exp / 10);
    }
}
  1. 生成proxy
public static void main(String[] args) {
        IGame game = new GameImpl();  
                // 动态生成代理类,
        IGame proxy = (IGame) Proxy.newProxyInstance(  
                game.getClass().getClassLoader(),  
                game.getClass().getInterfaces(),  
                new ReadyInvocationHandler(game));  
        // 调用时机
        proxy.play();  
        System.out.println(proxy.getLevel(50));
    }
  1. 代理中调用过程
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

public class ReadyInvocationHandler implements InvocationHandler {  
    //IGame接口的实现类
    private Object game = null;  
  
    public ReadyInvocationHandler(Object realSubject) {  
        this.game = realSubject;  
    }  
  
    public Object invoke(Object proxy, Method m, Object[] args) {  
        Object result = null;  
        try {  
            if ("play".contains(method.getName())) {

                    System.out.println(proxy.getClass().getSimpleName());
                    System.out.println("游戏开始前,开启辅助插件");
                    // 真正的调用play的地方
                    result = method.invoke(game, args);
              }else if ("getLevel".equals(method.getName())){
                    System.out.println("开始计算等级");
                    // 做点小改动啥的
                    int exp = (int) args[0];
                    args[0] = exp + 20;
                    result = method.invoke(game, args);
              } 
        } catch (Exception ex) {  
            System.exit(1);  
        }  
        return result;  
    }  
}
  1. 输出结果
$Proxy0
游戏开始前,开启辅助插件
开始游戏!
开始计算等级
7

动态代理的实现原理

和静态代理的不同点是无需我们手动或依靠工具来生成代理类的代码,动态代理由JDK为我们自动实现代理对象,这又是什么原理呢?
比如现在想为RealSubject这个类创建一个动态代理对象,JDK主要会做以下工作:

1.   获取 RealSubject上的所有接口列表;
2.   确定要生成的代理类的类名,默认为:com.sun.proxy.$ProxyXXXX;
3.   根据需要实现的接口信息,在代码中动态创建 该Proxy类的字节码;
4 .  将对应的字节码转换为对应的class 对象;
5.   创建InvocationHandler 实例handler,用来处理Proxy所有方法调用;
6.   Proxy 的class对象 以创建的handler对象为参数,实例化一个proxy对象

参考Java动态代理机制详解(JDK 和CGLIB,Javassist,ASM)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,222评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,455评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,720评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,568评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,696评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,879评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,028评论 3 409
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,773评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,220评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,550评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,697评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,360评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,002评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,782评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,010评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,433评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,587评论 2 350

推荐阅读更多精彩内容

  • 全文 1483 字 | 建议阅读 3 分钟 如果你相信我写的题目,那你就错了。如果你不相信我写的题目,那我就错了。...
    三风mickjoust阅读 269评论 0 28
  • 天,依旧阴沉着。 独自一人徘徊在十字路口的交叉处,不知道向左,还是向右。多想,就这样一直走下去。没有烦恼,没有忧虑...
    章鱼去哪儿阅读 337评论 0 0
  • 本人情况:普通二本,专业没有通过评估。没有任何的背景关系。在学校学习很努力,因为只能靠自己,也没有抱怨,知道抱怨...
    X_PLAN阅读 6,196评论 28 15
  • 给定一个包含 n 个整数的数组 nums,判断 nums 中是否存在三个元素 a,b,c ,使得 a + b + ...
    小白学编程阅读 189评论 0 0
  • 向导功能 假如你计算机上在过去已经有安装过 IntelliJ IDEA 14 的版本,且你在卸载 IntelliJ...
    super1981阅读 277评论 0 0