一、代理模式概述

  1、代理模式的核心作用

    (1)通过代理,控制对象的访问;

    (2)可以详细的控制访问某个(某类)对象的方法,在调用这个方法前做前置处理,调用这个方法后做后置处理。(AOP的微实现)

        代理模式是 AOP(Aspect Oriented Programming 面向切面编程)的核心实现机制。

  2、代理模式的核心角色

    (1)抽象角色

      定义代理角色和真实角色的公共对外方法。

    (2)真实角色

      实现抽象角色,定义真实角色所需要实现的业务逻辑,供代理角色调用。

      关注真正的业务逻辑。

    (3)代理角色

      实现抽象角色,是真实角色的代理,通过真实角色的业务逻辑辅方法来实现抽象方法,并附加自己的操作。

      将统一的流程控制放在代理角色中处理!

  3、代理模式的使用场景

    (1)安全代理: 屏蔽对真实角色的访问;

    (2)通过代理类处理远程方法调用(RMI Remote Method Invocation 远程方法调用);

    (3)延迟加载:先加载轻量级的代理对象,真正需要再加载真实对象。

  4、代理模式的分类

    (1)静态代理(静态定义代理类)

    (2)动态代理(动态生成代理类)

      ① JDK自带的动态代理

      ② javassist字节码操作库实现

      ③ CGLIB

        CGLIB是一个功能强大,高性能的代码生成包。它为没有实现接口的类提供代理,为JDK的动态代理提供了很好的补充。

        通常可以使用Java的动态代理创建代理,但当要代理的类没有实现接口或者为了更好的性能,CGLIB是一个好的选择。

      ④ ASM底层使用指令,可维护性较差

        ASM是一个java字节码操纵框架,它能被用来动态生成类或者增强既有类的功能。

二、代理模式之静态代理

  示例代码场景:周杰伦开演唱会时,面谈、签合同、订票等操作都是由周杰伦的经纪人来完成,唱歌部分由他本人完成,最后收款也是由经纪人来完成。

  (1)创建抽象角色

 /**
* 抽象角色
* @author CL
*
*/
public interface Star {
/**
* 面谈
*/
void confer();
/**
* 签合同
*/
void signConstract();
/**
* 订票
*/
void bookTicket();
/**
* 唱歌
*/
void sing();
/**
* 收款
*/
void collectMoney();
}

  (2)定义真实角色

 /**
* 真实身份(相当于例子中的周杰伦本人)
* @author CL
*
*/
public class RealStar implements Star { @Override
public void confer() {
System.out.println("周杰伦去面谈");
} @Override
public void signConstract() {
System.out.println("周杰伦去签合同");
} @Override
public void bookTicket() {
System.out.println("周杰伦去订票");
} @Override
public void sing() {
System.out.println("周杰伦去唱歌");
} @Override
public void collectMoney() {
System.out.println("周杰伦去收款");
} }

  (3)定义代理角色

 /**
* 代理身份(相当于例子中的周杰伦的经纪人)
* @author CL
*
*/
public class ProxyStar implements Star {
private Star star; public ProxyStar(Star star) {
this.star = star;
} @Override
public void confer() {
System.out.println("经纪人去面谈");
} @Override
public void signConstract() {
System.out.println("经纪人去签合同");
} @Override
public void bookTicket() {
System.out.println("经纪人去订票");
} @Override
public void sing() {
star.sing();
} @Override
public void collectMoney() {
System.out.println("经纪人去收款");
} }

  (4)测试

 /**
* 客户端
* 只和经纪人联系
* @author CL
*
*/
public class Client { public static void main(String[] args) {
Star real = new RealStar();
Star proxy = new ProxyStar(real); //面谈
proxy.confer();
//签合同
proxy.signConstract();
//订票
proxy.bookTicket();
//唱歌
proxy.sing();
//收款
proxy.collectMoney();
}
}

  控制台输出:

经纪人去面谈
经纪人去签合同
经纪人去订票
周杰伦去唱歌
经纪人收款

三、代理模式之动态代理

  JDK自带的动态代理:

    ① 在客户端,java.lang.reflect.Proxy

     作用:动态生成代理类和对象

    ② 实现java.lang.reflect.InvocationHandler(处理器接口)

     可以通过invoke方法实现对真实角色的代理访问

     每次通过Proxy生成代理类对象时都要指定对应的处理器对象

  (1)创建抽象角色

 public interface Star {
/**
* 面谈
*/
void confer();
/**
* 签合同
*/
void signConstract();
/**
* 订票
*/
void bookTicket();
/**
* 唱歌
*/
void sing();
/**
* 收款
*/
void collectMoney();
}

  (2)创建真实角色

 /**
* 真实身份(相当于例子中的周杰伦本人)
* @author CL
*
*/
public class RealStar implements Star { @Override
public void confer() {
System.out.println("周杰伦去面谈");
} @Override
public void signConstract() {
System.out.println("周杰伦去签合同");
} @Override
public void bookTicket() {
System.out.println("周杰伦去订票");
} @Override
public void sing() {
System.out.println("周杰伦去唱歌");
} @Override
public void collectMoney() {
System.out.println("周杰伦去收款");
} }

  (3)创建代理角色的处理器

 import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method; /**
* 代理角色的处理类
* jdk自带的动态代理
* @author CL
*
*/
public class StarHandler implements InvocationHandler {
private Star realStar; public StarHandler(Star realStar) {
this.realStar = realStar;
} //通过invoke方法实现对真实角色的代理访问
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
Object obj = null; System.out.print("在执行真实角色的方法之前的处理,比如-->");
System.out.println("面谈,签合同,订票"); if (method.getName().equals("sing")) {
obj = method.invoke(realStar, args);
} System.out.print("在执行真实角色的方法之后的处理,比如-->");
System.out.println("收款"); return obj;
} }

  (4)测试

 import java.lang.reflect.Proxy;

 /**
* 测试JDK自带的动态代理
* @author CL
*
*/
public class Client { public static void main(String[] args) {
Star realStar = new RealStar(); StarHandler handler = new StarHandler(realStar); Star proxy = (Star) Proxy.newProxyInstance(ClassLoader.getSystemClassLoader(),new Class[]{Star.class}, handler); proxy.sing();
}
}

  控制台输出:

在执行真实角色的方法之前的处理,比如-->面谈,签合同,订票
周杰伦去唱歌
在执行真实角色的方法之后的处理,比如-->收款

GOF23设计模式之代理模式(proxy)的更多相关文章

  1. GOF23设计模式之代理模式

    GOF23设计模式之代理模式 核心作用:通过代理,控制对对象的访问.可以详细控制访问某个(某类)对象的方法,在调用这个方法前做前置处理,调用这个方法后做后置处理(即:AOP的微观实现) AOP(Asp ...

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

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

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

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

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

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

  5. 【GOF23设计模式】代理模式

    来源:http://www.bjsxt.com/ 一.[GOF23设计模式]_代理模式.静态代理 package com.test.proxy.staticProxy; public interfac ...

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

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

  7. [设计模式] 12 代理模式 proxy

    在GOF的<设计模式:可复用面向对象软件的基础>一书中对代理模式是这样说的:为其他对象提供一种代理以控制对这个对象的访问.结合上面的游戏代理的例子和下面的图,我们来进行分析一下.以前你是这 ...

  8. c#设计模式之代理模式(Proxy Pattern)

    引言 代理这个词语,大家在现实世界已经频繁的接触过,例如火车站代理售票点,因为这些代理售票点的存在,我们不必要去火车站的售票处就可以查询或者取到火车票.代理点本身是没有能力生产车票的,我们在代理处享受 ...

  9. 设计模式三: 代理模式(Proxy) -- JDK的实现方式

    简介 代理模式属于行为型模式的一种, 控制对其他对象的访问, 起到中介作用. 代理模式核心角色: 真实角色,代理角色; 按实现方式不同分为静态代理和动态代理两种; 意图 控制对其它对象的访问. 类图 ...

随机推荐

  1. Netty实例几则

    Netty是基于JDK NIO的网络框架 简化了NIO编程, 不用程序自己维护selector, 将网络通信和数据处理的部分做了分离 多用于做底层的数据通信, 心跳检测(keepalived) 1. ...

  2. 011PHP基础知识——运算符(四)

    <?php /** * 连接运算符: . 连接2个参数生成新的字符串: */ /*$str="中国"; $bbs="bbs.blog.com"; $new ...

  3. MD5加密(java和c#)

    java代码 public static String md5(String str) { try { MessageDigest md = MessageDigest.getInstance(&qu ...

  4. 内存保护机制及绕过方法——利用Ret2Libc绕过DEP之VirtualProtect函数

    利用Ret2Libc绕过DEP之VirtualProtect函数 ⑴.  原理分析: i.相关概念: VirtualProtect()函数: BOOL WINAPI VirtualProtect( _ ...

  5. springcloud- FeginClient 调用统一拦截添加请求头 RequestInterceptor ,被调用服务获取请求头

    使用场景: 在springcloud中通过Fegin调用远端RestApi的时候,经常需要传递一些参数信息到被调用服务中去,比如从A服务调用B服务的时候, 需要将当前用户信息传递到B调用的服务中去,我 ...

  6. APUE学习笔记——8.1-8.4 进程基础

    进程ID 1 进程id是唯一的.(不会有进程id一样的两个进程) 2进程id是可复用的,一个进程销毁后,它的id号可以被新的进程使用.但是Unix采用了延迟复用的算法,也就是进程   销毁后它的id不 ...

  7. Unity3D使用溶解技术解决障碍物遮挡

    笔者介绍:姜雪伟,IT公司技术合伙人,IT高级讲师,CSDN社区专家,特邀编辑,畅销书作者,已出版书籍:<手把手教你架构3D游戏引擎>电子工业出版社和<Unity3D实战核心技术详解 ...

  8. String.prototype.getParm

    String.prototype.getParms=function(name){ var reg = new RegExp('(^|&)' + name + '=([^&]*)(&a ...

  9. 在FP与DDD的道路上越走越远

    托辞 最近一口气读了若干本FP方面的书,可人是越来越懒了,要整理个什么东西却也没有那个精力,所以简单扔几张图,算是给自己一个提醒吧. 在此期间,我用并不纯熟的Scala和Groovy练了一把手.虽然只 ...

  10. sync or async connect redis in golang

    Head of head 在golang的整个生态里,redis client lib全部都使用多连接或者连接池.这是让人难以理解的,所以我和xiaofei一起写了一个同时支持同步和异步的redis ...