代理

1.代理原理

 利用动态代理可以在运行时创建一个实现了一组给定接口的新类。假设有一个表示接口的Class对象,它的确切类型在编译时无法知道(Proxy.newInstance()方法ClassLoader参数为null的情况)。所以要想构造一个实现这些接口的类,需要使用newInstance()方法反射找出这个类的构造器。但是,不能实例化一个接口,需要在程序处于运行状态时定义一个新类
 代理机制可以满足上述需求,代理类可以在运行时创建全新的类,这样的代理类能够实现指定的接口

2.代理场景

代理这种设计模式是通过不直接访问被代理对象的方式,而访问代理对象的方法。这个就好比商户---->明星经纪人(代理)---->明星这种模式。我们可以不通过直接与明星对话的情况下,而通过明星经纪人(代理)与其产生间接对话
设计模式中有一个设计原则是开闭原则,是说对修改关闭对扩展开放,我们在工作中有时会接手很多前人的代码,里面代码逻辑让人摸不着头脑,这时就很难去下手修改代码,那么这时我们就可以通过代理对类进行增强

3.代理实现

在java中规定,要想产生一个对象的代理对象,那么这个对象必须要有一个接口

public interface Person {
 
     String sing(String name);

     String dance(String name);
 }

public class LiuDeHua implements Person {
 
     public String sing(String name){
         System.out.println("刘德华唱"+name+"歌!!");
         return "歌唱完了,谢谢大家!";
     }
     
     public String dance(String name){
         System.out.println("刘德华跳"+name+"舞!!");
         return "舞跳完了,多谢各位观众!";
     }
 }

 要想创建一个代理对象,需要使用Proxy类的newProxyInstance方法。这个方法需要三个参数:
ClassLoader loader:指定一个动态加载代理类的类加载器(可为null)
Class<?>[] interfaces:指明被代理类实现的接口,之后我们通过拼接字节码生成的类才能知道调用哪些方法。
InvocationHandler h:这是一个方法委托类,我们通过代理调用被代理类的方法时,就可以将方法名和方法参数都委托给这个委托类。

public class LiuDeHuaProxy {
 
     //设计一个类变量记住代理类要代理的目标对象
     private Person ldh = new LiuDeHua();
     
     /**
     * 设计一个方法生成代理对象
     * @Method: getProxy
     * @Description: 这个方法返回刘德华的代理对象:Person person = LiuDeHuaProxy.getProxy();//得到一个代理对象
     * @return 某个对象的代理对象
     */ 
     public Person getProxy() {
         //使用Proxy.newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)返回某个对象的代理对象
         return (Person) Proxy.newProxyInstance(LiuDeHuaProxy.class
                 .getClassLoader(), ldh.getClass().getInterfaces(),
                 new InvocationHandler() {
                     /**
                      * InvocationHandler接口只定义了一个invoke方法,因此对于这样的接口,我们不用单独去定义一个类来实现该接口,
                      * 而是直接使用一个匿名内部类来实现该接口,new InvocationHandler() {}就是针对InvocationHandler接口的匿名实现类
                      */
                     /**
                      * 在invoke方法编码指定返回的代理对象干的工作
                      * proxy : 把代理对象自己传递进来 
                      * method:把代理对象当前调用的方法传递进来 
                      * args:把方法参数传递进来
                      * 
                      * 当调用代理对象的person.sing("冰雨");或者 person.dance("江南style");方法时,
                      * 实际上执行的都是invoke方法里面的代码,
                      * 因此我们可以在invoke方法中使用method.getName()就可以知道当前调用的是代理对象的哪个方法
                      */
                     @Override
                     public Object invoke(Object proxy, Method method,
                             Object[] args) throws Throwable {
                         //如果调用的是代理对象的sing方法
                         if (method.getName().equals("sing")) {
                             System.out.println("我是他的经纪人,要找他唱歌得先给十万块钱!!");
                             //已经给钱了,经纪人自己不会唱歌,就只能找刘德华去唱歌!
                             return method.invoke(ldh, args); //代理对象调用真实目标对象的sing方法去处理用户请求
                         }
                         //如果调用的是代理对象的dance方法
                         if (method.getName().equals("dance")) {
                             System.out.println("我是他的经纪人,要找他跳舞得先给二十万块钱!!");
                             //已经给钱了,经纪人自己不会唱歌,就只能找刘德华去跳舞!
                             return method.invoke(ldh, args);//代理对象调用真实目标对象的dance方法去处理用户请求
                         }
 
                         return null;
                     }
                 });
     }
 }
public class ProxyTest {     
      public static void main(String[] args) {
          LiuDeHuaProxy proxy = new LiuDeHuaProxy();
          //获得代理对象
          Person p = proxy.getProxy();
         //调用代理对象的sing方法
         String retValue = p.sing("冰雨");
         System.out.println(retValue);
         //调用代理对象的dance方法
         String value = p.dance("江南style");
         System.out.println(value);
     }
 }

对于特定的类的加载器预设的一组接口来说,只能有一个代理类。也就是说,如果使用同一个类加载器接口数组调用两次newProInstance方法的话,那么只能得到同一个类得两个对象

 可以通过调用Proxy类中的isProxyClass方法检测一个特定的Class对象是否代表一个代理类。

4.动态代理和静态代理的区别

静态代理:

public class Proxy implements Person {
     private LiuDeHua mLDH;
      
     public Proxy(LiudeHua ldh){
      this.mLDH = ldh;
      }
     public String sing(String name){
         mLDH.sing(name);
     }
     
     public String dance(String name){
         mLDH.dance(name);
     }
 }

 根据加载被代理类时机不同,将代理分为静态代理动态代理。如果在代码编译时确定被代理的类是哪一个,那么就可以直接使用静态代理;如果不能确定,那么可以使用类的动态加载机制,在代码运行期间加载被代理的类这就是动态代理,比如RPC框架和Spring AOP机制。

 对于静态代理方式代理类需要实现和被代理类相同的接口;对于动态代理代理类不需要显示地实现被代理类所实现的接口

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

推荐阅读更多精彩内容

  • 一、基本概念 1.什么是代理? 在阐述JDK动态代理之前,我们很有必要先来弄明白代理的概念。代理这个词本身并不是计...
    小李弹花阅读 16,422评论 2 40
  • 整体Retrofit内容如下: 1、Retrofit解析1之前哨站——理解RESTful 2、Retrofit解析...
    隔壁老李头阅读 3,210评论 2 10
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,076评论 1 32
  • 抽屉 曾经有一个抽屉 打开来里边有一个饼干盒 但盒里没有饼干 有电池,有证件,有照片,有体温计 还有一副老花镜 它...
    会咬人的扑克牌阅读 154评论 0 0
  • 就在今晚,我知道了我的一个大学学长车祸去世的消息。和他并不很熟,甚至连点头之交都算不上,因为他并不认识我。可是依然...
    苔花小姐阅读 394评论 0 0