代理模式定义:为另一个对象提供一个替身或者占位符以控制对这个对象的访问。---《Head First 设计模式》

代理模式换句话说就是给某一个对象创建一个代理对象,由这个代理对象控制对原对象的引用,并且创建这个代理对象后可以在调用原对象时增加一些额外操作。

下面就由一张代理原理图说明:

1.Subject抽象主题,Proxy和RealSubject都要实现的接口。这允许任何客户都可以像处理RealSubject对象一样地处理Proxy.

2.Proxy:代理类,必须持有所代理对象的引用。

3.RealSubject被代理的类,真正做事的对象。Proxy会控制对RealSubject的访问。

举一个栗子:我们有一个看电影的接口,此时有一个只有看电影的类、还有一个代理商,代理了这部电影,并在看电影开始和结束都加入了广告。

看电影接口:

/**
* @author monkjavaer
* @date 2018/09/05 22:18
*/
public interface Subject { void seeMovie();
}

真正的电影:

/**
* @author monkjavaer
* @date 2018/09/05 22:19
*/
public class RealSubject implements Subject{ @Override
public void seeMovie() {
System.out.println("RealSubject:看电影");
}
}

代理商:

/**
* 代理类
* @author monkjavaer
* @date 2018/09/05 22:20
*/
public class Proxy implements Subject { private Subject subject; public Proxy(Subject subject) {
this.subject = subject;
} @Override
public void seeMovie() {
System.out.println("Proxy:加广告");
subject.seeMovie();
System.out.println("Proxy:加广告");
}
}

测试:

/**
* 测试
* @author monkjavaer
* @date 2018/09/05 22:22
*/
public class ProxyTest {
public static void main(String[] args) { Proxy proxy = new Proxy(new RealSubject());
proxy.seeMovie(); }
}

  结果:

  Proxy:加广告
  RealSubject:看电影
  Proxy:加广告

上面就是我们传说中的静态代理。

好了,说了这么多,总结静态代理123。。。

优点: RealSubject通过Proxy实现了功能的增强。 在没有Proxy的时候,RealSubject依然可以执行。

缺点: Proxy要实现和RealSubject同样的接口Subject,Proxy只能为Subject进行代理;如果有很多接口的实现类需要代理,那么就需要很多的代理类。

嗯,,,既然静态代理有这个缺点,那我们啷个办呢。。。。。。

下面我们就引入动态代理。

jdk动态代理:java动态代理位于java.lang.reflect这个包下。其实现原理就是通过反射获取构造函数对象并生成代理类实例。

java Proxy 源码中有这样的一段描述:

/*
* Look up or generate the designated proxy class.
*/
Class<?> cl = getProxyClass0(loader, intfs); /*
* Invoke its constructor with the designated invocation handler.
*/
try {
if (sm != null) {
checkNewProxyPermission(Reflection.getCallerClass(), cl);
} final Constructor<?> cons = cl.getConstructor(constructorParams);
final InvocationHandler ih = h;
if (!Modifier.isPublic(cl.getModifiers())) {
AccessController.doPrivileged(new PrivilegedAction<Void>() {
public Void run() {
cons.setAccessible(true);
return null;
}
});
}
return cons.newInstance(new Object[]{h});
} catch (IllegalAccessException|InstantiationException e) {
throw new InternalError(e.toString(), e);
}

JDK动态代理的实现步骤:

1.创建一个实现接口InvocationHandler的类,它必须实现invoke方法

2.创建被代理的类以及接口

3.通过Proxy的静态方法newProxyInstance(ClassLoaderloader, Class[] interfaces, InvocationHandler h)创建一个代理类

4.通过代理类调用方法

下面是具体的java代码实现:

场景:小明吃饭前后都有人帮忙做一些事情。。。

/**
* 人
*/
public interface People { void eat();
}

小明:

public class XiaoMing implements People {
@Override
public void eat() {
System.out.println("小明吃红烧肉");
}
}

代理类:

/**
* 代理类
* @author monkjavaer
* @date 2018/8/31 15:39
*/
public class InvocationHandlerImpl implements InvocationHandler { /**
* 代理的对象
*/
private Object object; public InvocationHandlerImpl(Object object) {
this.object = object;
} /**
*
* @param proxy JDK动态生成的最终代理对象
* @param method 调用真实对象的某个方法的Method对象
* @param args 调用真实对象某个方法时接受的参数
* @return 代理对象
* @throws Throwable
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("before:把饭做好"); //当代理对象调用真实对象的方法时,其会自动的跳转到代理对象关联的handler对象的invoke方法来进行调用
Object invokeObject = method.invoke(object, args); System.out.println("after:洗碗");
return invokeObject;
}
}

测试:

/**
* 小明吃饭
* @author monkjavaer
* @date 2018/8/31 15:53
*/
public class ProxyTest {
public static void main(String[] args) {
People people = new XiaoMing();
InvocationHandlerImpl handler1 = new InvocationHandlerImpl(people);
//类加载器
ClassLoader classLoader1 = people.getClass().getClassLoader();
//得到全部的接口
Class[] interfaces1 = people.getClass().getInterfaces();
//返回代理类的一个实例
People peopleProxy = (People) Proxy.newProxyInstance(classLoader1, interfaces1, handler1); peopleProxy.eat();
}
}

 根据上面的jdk动态代理我们得知: 

jdk动态代理有局限,必须针对接口,如果没有接口像上面的例子People、目标没有实现接口那么这个办法就GG了。

这时我们就需要用到另一种代理Cglib动态代理。预知Cglib请听下回分解。。。

夜话JAVA设计模式之代理模式(Proxy)的更多相关文章

  1. Java设计模式之代理模式(静态代理和JDK、CGLib动态代理)以及应用场景

    我做了个例子 ,需要可以下载源码:代理模式 1.前言: Spring 的AOP 面向切面编程,是通过动态代理实现的, 由两部分组成:(a) 如果有接口的话 通过 JDK 接口级别的代理 (b) 如果没 ...

  2. java设计模式6——代理模式

    java设计模式6--代理模式 1.代理模式介绍: 1.1.为什么要学习代理模式?因为这就是Spring Aop的底层!(SpringAop 和 SpringMvc) 1.2.代理模式的分类: 静态代 ...

  3. 乐在其中设计模式(C#) - 代理模式(Proxy Pattern)

    原文:乐在其中设计模式(C#) - 代理模式(Proxy Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 代理模式(Proxy Pattern) 作者:webabcd 介绍 为 ...

  4. Java设计模式:代理模式(转)

    代理(Proxy)是一种设计模式,提供了对目标对象另外的访问方式;即通过代理对象访问目标对象.这样做的好处是:可以在目标对象实现的基础上,增强额外的功能操作,即扩展目标对象的功能.这里使用到编程中的一 ...

  5. 大熊君说说JS与设计模式之------代理模式Proxy

    一,总体概要 1,笔者浅谈 当我们浏览网页时,网页中的图片有时不会立即展示出来,这就是通过虚拟代理来替代了真实的图片,而代理存储了真实图片的路径和尺寸,这就是代理方式的一种. 代理模式是比较有用途的一 ...

  6. 二十四种设计模式:代理模式(Proxy Pattern)

    代理模式(Proxy Pattern) 介绍为其他对象提供一个代理以控制对这个对象的访问. 示例有一个Message实体类,某对象对它的操作有Insert()和Get()方法,用一个代理来控制对这个对 ...

  7. Java基础-设计模式之-代理模式Proxy

    代理模式是对象的结构模式.代理模式给某一个对象提供一个代理对象,并由代理对象控制对原对象的引用. 代理模式是常用的Java 设计模式,它的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理 ...

  8. Java设计模式之代理模式(Proxy)

    前言: 最近在研究Retrofit开源框架的时候,其主要核心代码是通过注解标示参数,动态代理模式实现具体接口,反射机制进行参数解析,最终实现发送请求.其实之前在学习Xutils源码的时候,Xutils ...

  9. java设计模式之八代理模式(Proxy)

    其实每个模式名称就表明了该模式的作用,代理模式就是多一个代理类出来,替原对象进行一些操作,比如我们在租房子的时候回去找中介,为什么呢?因为你对该地区房屋的信息掌握的不够全面,希望找一个更熟悉的人去帮你 ...

随机推荐

  1. c++ rand && srand 函数

    RAND: 结构:rand()     注:rand()不需要参数,它会根据种子返回一个从0到最大随机数的任意整数. 作用:生成一个随机数. 头文件:#include <cstdlib> ...

  2. ASP.NET MVC5 之 客户端实现文件的下载

    MVC 实现下载功能主要借助于 File 属性: //下载文件接口 public ActionResult GetTrackTempIsc(ICSModels icsModels) { bool fl ...

  3. 获取Sprite上某一个点的透明度

    转载[ http://www.cnblogs.com/Androider123/p/3795050.html] 本篇文章主要讲一下怎么做一个不规则的按钮,比如如下图的八卦,点击绿色和点击红色部分,需要 ...

  4. RabbitMQ指南之四:路由(Routing)和直连交换机(Direct Exchange)

    在上一章中,我们构建了一个简单的日志系统,我们可以把消息广播给很多的消费者.在本章中我们将增加一个特性:我们可以订阅这些信息中的一些信息.例如,我们希望只将error级别的错误存储到硬盘中,同时可以将 ...

  5. log4j2异步日志解读(一)AsyncAppender

    log4j.logback.log4j2 历史和关系,我们就在这里不展开讲了.直接上干货,log4j2突出于其他日志的优势,异步日志实现. 看一个东西,首先看官网文档 ,因为前面文章已经讲解了disr ...

  6. Spring:(三) --常见数据源及声明式事务配置

    Spring自带了一组数据访问框架,集成了多种数据访问技术.无论我们是直接通过 JDBC 还是像Hibernate或Mybatis那样的框架实现数据持久化,Spring都可以为我们消除持久化代码中那些 ...

  7. Objective-C设计模式——中介者Mediator(对象去耦)

    中介者模式 中介者模式很好的诠释了迪米特法则,任意两个不相关的对象之间如果需要关联,那么需要通过第三个类来进行.中介者就是把一组对象进行封装,屏蔽了类之间的交互细节,使不同的类直接不需要持有对方引用也 ...

  8. how to do a mass update in Laravel5 ( 在Laravel 5里面怎么做大量数据更新 )

    Today, I had spent 3 hours to fix one problem, The old program has a bug, originally, when a user pr ...

  9. Laravel5.1学习笔记22 Eloquent 调整修改

    Eloquent: Mutators Introduction Accessors & Mutators Date Mutators Attribute Casting Introductio ...

  10. maven idea

    写在前面的话:此篇文章教程是在IntelliJ IDEA中搭建的maven项目.(建议eclipse党快点转IDEA吧,IDEA大法好.逃… 1.maven的安装 前往Apache Maven官网点击 ...