前言:

  代理模式作为常见的设计模式之一,在项目开发中不可或缺。本文就尝试着揭开代理的神秘面纱,也欢迎各路人批评指正!

1.如何实现代理:

【假设有个关于汽车移动(move)的计时需求】
设计:Moveable接口,一个Car的实现类;两个代理CarTimer,TimeHandler.UML图如下:


1)继承

 package com.gdufe.proxy;

 import java.util.Random;

 public class CarTimer extends Car {

     @Override
public void move() {
long start=System.currentTimeMillis();
super.move(); //调用父类的move()方法
try{
Thread.sleep(new Random().nextInt(10000));
}catch(Exception e){
e.printStackTrace();
}
long end=System.currentTimeMillis();
System.out.println("I'm time machine.Time for moving:"+(end-start));
}
}


2)组合

 package com.gdufe.proxy;

 import java.util.Random;

 public class TimeHandler implements Moveable {
private Moveable m;
public TimeHandler(Moveable m) {
this.m = m;
}
@Override
public void move() {
long start=System.currentTimeMillis();
m.move();
try{
Thread.sleep(new Random().nextInt(10000));
}catch(Exception e){
e.printStackTrace();
}
long end=System.currentTimeMillis();
System.out.println("I'm time machine.Time for moving:"+(end-start));
} }

客户端代码:

 package com.gdufe.proxy;

 public class Client {
public static void main(String[] args) {
System.out.println("继承实现代理:");
new CarTimer().move();
System.out.println("组合实现代理:");
new TimeHandler(new Car()).move();
}
}


输出结果:

继承实现代理:
Car moving...
I'm time machine.Time for moving:7080
组合实现代理:
Car moving...
I'm time machine.Time for moving:5169

分析:从上述例子实现当中,我们第一感觉自然是分不出两种代理的实现方式孰优孰劣。且继续往下面看。

2.灵活代理-接口切换

【假设现在特殊需求不确定:“汽车移动之前先往左还是先往右”】
很明显,我们此时若使用继承的方式实现代理,则后续很不容易维护,而且会形成臃肿的继承链;但使用接口的方式我们发现仅需要两个代理类:TurnLeft,TurnRight。而且,不管后续需求如何都只需要做简单的调整。UML图如下:

----------

TurnLeft.java

 package com.gdufe.proxy;

 public class TurnLeft implements Moveable {
private Moveable m;
public TurnLeft(Moveable m) {
this.m = m;
}
@Override
public void move() {
System.out.println("turn left...");
m.move();
} }

TurnRight.java

 package com.gdufe.proxy;

 public class TurnRight implements Moveable {

     private Moveable m;
public TurnRight(Moveable m) {
this.m = m;
}
@Override
public void move() {
System.out.println("turn right");
m.move();
} }

客户端代码:

 package com.gdufe.proxy;

 public class Client0 {

     /**
* @param args
*/
public static void main(String[] args) {
System.out.println("Turn right,then left before moving:");
test1(); System.out.println("Turn left,then right before moving:");
test2();
}
//对接口的实现内外包装
private static void test1() {
Car car = new Car();
Moveable m1 = new TurnLeft(car);
Moveable m2 = new TurnRight(m1);
m2.move();
}
public static void test2(){
Car car = new Car();
Moveable m1 = new TurnRight(car);
Moveable m2 = new TurnLeft(m1);
m2.move();
} }

输出结果:

Turn right,then left before moving:
turn right
turn left...
Car moving...
Turn left,then right before moving:
turn left...
turn right
Car moving...

========================

3.动态代理:

其实,不管是继承还是组合的方式,我们上面实现的都仅仅是“静态代理”,也是我们平时用的比较多的。现在,我们开始聊聊Java的神器---“动态代理”。
【假设现在需要实现一个“万能”的日志工具,即不管对任何类的任何方法都可以动态对其进行日志操作】
例如:上面的例子中,请思考如何实现在汽车移动之前进行日志操作?
常规的静态代理方式当然可以实现,下面我们就利用Java中的Proxy类进行实现。UML图如下:

添加关键类:LogHandler.java

 package com.gdufe.proxy;

 import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method; public class LogHandler implements InvocationHandler { //被代理对象
private Object proxied; public LogHandler(Object proxied) {
this.proxied = proxied;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
System.out.println("I'm log machine.");
return method.invoke(proxied);
} }

客户端代码:

 package com.gdufe.proxy;

 import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy; public class Client1 { /**
* @param args
*/
public static void main(String[] args) { //单独创建Moveable接口对象
Moveable m=null;
m = new Car();
m = new TimeHandler(m); //初始化LogHandler实现的InvacationHandler接口
InvocationHandler h = new LogHandler(m);
m = (Moveable) Proxy.newProxyInstance(
Moveable.class.getClassLoader(),
new Class[] {Moveable.class},h); //m在动态代理处理完后,move()方法已被修改
m.move();
} }

输出结果:

I'm log machine.
Car moving...
I'm time machine.Time for moving:110

分析:

  上述的实现代码对于刚接触Java的朋友来说估计比较费解。要理解其过程,至少对java的反射机制有一定的理解。看穿了的话,其实动态代理的关键环节,就在newProxyInstance()操作上。代码实现的关键步骤:

Step1:初始化被代理的对象(如上图中的TimeHandler);

Step2:创建一个实现了InvocationHandler接口的类,在invoke()方法进行你希望执行的代理操作(如上图的LogHandler);

Step3:将通过Proxy拿到的新对象赋给最终要实现的接口,最后调用该接口方法(如代码中的“m.move()”)。

---------------------------------------------------------------------

有朋友可能犯迷糊了,动态代理的内部实现过程呢?Proxy是怎样一步一步识别到Car的呢?

请看图:

注意:$Proxy类是虚拟存在的,在Java API中是找不到的。也就是说,它只存在于中间过程。所以,为方便大家理解就加上了。)

  
  那么,到底动态代理适用于什么情形呢?从上面汽车的简单日志实例也许还难以看出。下面我们再引入一个测试。
【假设现在增加一个司机(Driver)类,其实现了Speakable接口;很明显他跟汽车没有很直接的关联,那么现在我们利用动态代理的方式将上面的LogHandler加到司机的speak()方法上】
----------
客户端代码:

 package com.gdufe.proxy;

 import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy; public class Client2 { /**
* @param args
*/
public static void main(String[] args) {
Moveable m=null;
m = new Car(); Speakable s =null;
s = new Driver(); InvocationHandler h = null;
h=new LogHandler(m);
m = (Moveable) Proxy.newProxyInstance(Moveable.class.getClassLoader(),new Class[] {Moveable.class},h);
m.move(); //司机被代理
h = new LogHandler(s);
s = (Speakable)Proxy.newProxyInstance(Speakable.class.getClassLoader(),new Class[]{Speakable.class}, h);
s.speak();
} }

输出结果:

I'm log machine.
Car moving...
I'm log machine.
Driver speak...

  在汽车move()跟司机speak()之前,都自动实现LogHandler操作!

-----------------------------------------------------

后语:

通过上述例子,总结动态代理优点:
·适用任何类的任何方法;
·使用灵活,可随时将代理工具类加入或抽出。

  动态代理是Java语言的精华所在,很多的开发框架都是基于其内部原理。本人目前对动态代理的理解也仅限于此,欢迎对Java有深度学识的朋友拍砖,谢谢~

神秘代理-Proxy的更多相关文章

  1. 代理(Proxy)和反射(Reflection)

    前面的话 ES5和ES6致力于为开发者提供JS已有却不可调用的功能.例如在ES5出现以前,JS环境中的对象包含许多不可枚举和不可写的属性,但开发者不能定义自己的不可枚举或不可写属性,于是ES5引入了O ...

  2. 深度揭秘ES6代理Proxy

    最近在博客上看到关于ES6代理的文章都是一些关于如何使用Proxy的例子,很少有说明Proxy原理的文章,要知道只有真正掌握了一项技术的原理,才能够写出精妙绝伦的代码,所以我觉得有必要写一篇关于深刻揭 ...

  3. Webpack代理proxy配置,解决本地跨域调试问题,同时允许绑定host域名调试

    Webpack代理proxy配置,解决本地跨域调试问题,同时允许绑定host域名调试 会撸码的小马 关注 2018.05.29 17:30* 字数 212 阅读 1488评论 0喜欢 2 接到上一章, ...

  4. 浅谈Java代理一:JDK动态代理-Proxy.newProxyInstance

    浅谈Java代理一:JDK动态代理-Proxy.newProxyInstance java.lang.reflect.Proxy:该类用于动态生成代理类,只需传入目标接口.目标接口的类加载器以及Inv ...

  5. 初识代理——Proxy

    无处不在的模式——Proxy 最近在看<设计模式之禅>,看到代理模式这一章的时候,发现自己在写spring项目的时候其实很多时候都用到了代理,无论是依赖注入.AOP还是其他,可以说是无处不 ...

  6. JAVA设计模式-动态代理(Proxy)示例及说明

    在Mybatis源码解析,一步一步从浅入深(五):mapper节点的解析文章的最后部分,我们提到了动态代理的概念,下面我们就简单了解一下动态代理. 一,概念 代理设计模式的目的就是在不直接操作对象的前 ...

  7. JAVA设计模式-动态代理(Proxy)源码分析

    在文章:JAVA设计模式-动态代理(Proxy)示例及说明中,为动态代理设计模式举了一个小小的例子,那么这篇文章就来分析一下源码的实现. 一,Proxy.newProxyInstance方法 @Cal ...

  8. java动态代理--proxy&cglib

    大纲 代理 proxy cglib 小结 一.代理 为什么要用代理?其实就是希望不修改对象的情况下,增强对象. 静态代理: 静态代理模式,需要代理类和目标类实现同一接口,代理类的方法调用目标类的方法, ...

  9. 动态代理proxy与CGLib的区别

    什么是代理? 静态代理与动态代理 静态代理实例 JDK动态代理实例 CGLib 简介 CGLib 与JDK动态代理的区别 代理模式是Java中常见的一种模式,英文名字叫走Proxy或者Surrogat ...

随机推荐

  1. Is-A,Has-A,Use-A(转载)

    原文地址:http://blog.csdn.net/loveyou128144/article/details/4749576 而Is-A,Has-A,Use-A则是用来描述类与类之间关系的.简单的说 ...

  2. SpringMVC + Spring + MyBatis 整合 + Spring shrio + easyUI + 权限管理框架,带shrio session和shrio cache集群实现方案

    工作之余先来写了一个不算规范的简单架子 基于spring mvc + spring + mybatis + Spring shrio 基于redis的集群方案 系统权限部分,分成多个机构,其中每个机构 ...

  3. 千万pv大型web系统架构,学习从点滴开始

     架构,刚开始的解释是我从知乎上看到的.什么是架构?有人讲, 说架构并不是一 个很 悬 乎的 东西 , 实际 上就是一个架子 , 放一些 业务 和算法,跟我们的生活中的晾衣架很像.更抽象一点,说架构其 ...

  4. Android 手机卫士7--黑名单拦截

    1,黑名单数据库创建 三个字段(_id 自增长字段 phone 黑名单号码 mode 拦截类型) 创建表的sql语句 create table blacknumber (_id integer pri ...

  5. 微信小程序注册

    小程序是一种新的开放能力,可以在微信内被便捷地获取和传播,同时具有出色的使用体验.开发者可以根据平台提供的能力,快速地开发一个小程序. 开放内容包括: 1.开放注册范围:企业.政府.媒体.其他组织: ...

  6. 如何在webapp中做出原生的ios下拉菜单效果

    github:https://github.com/zhoushengmufc/iosselect webapp模仿ios下拉菜单 html下拉菜单select在安卓和IOS下表现不一样,iossel ...

  7. 初探React,将我们的View标签化

    前言 我之前喜欢玩一款游戏:全民飞机大战,而且有点痴迷其中,如果你想站在游戏的第一阶梯,便需要不断的练技术练装备,但是腾讯的游戏一般而言是有点恶心的,他会不断的出新飞机.新装备.新宠物,所以,很多时候 ...

  8. 使用WebRTC搭建前端视频聊天室——信令篇

    博客原文地址 建议看这篇之前先看一下使用WebRTC搭建前端视频聊天室——入门篇 如果需要搭建实例的话可以参照SkyRTC-demo:github地址 其中使用了两个库:SkyRTC(github地址 ...

  9. 微信Swift完整项目应用源码

    TSWeChat 中文说明 A WeChat alternative, written in Swift. 运行环境 Cocoapods 0.39.0 + iOS 8.0+ / Mac OS X 10 ...

  10. MySQL误操作后如何快速恢复数据

    基本上每个跟数据库打交道的程序员(当然也可能是你同事)都会碰一个问题,MySQL误操作后如何快速回滚?比如,delete一张表,忘加限制条件,整张表没了.假如这还是线上环境核心业务数据,那这事就闹大了 ...