【design pattern】代理模式
1. 设计模式分类
设计模式分为三大类:
创建型模式:工厂方法模式、抽象工厂模式、单例模式、建造者模式、原型模式;
结构型模式:适配器模式、装饰器模式、代理模式、外观模式、桥接模式、组合模式、享元模式;
行为型模式:策略模式、模板方法模式、观察者模式、迭代子模式、责任链模式、命令模式、备忘录模式、状态模式、访问者模式、中介者模式、解释器模式;
2. 代理模式
代理是基本的设计模式之一,提供额外的或不同的操作,代理通常充当中间人的角色,如果用“租房子”来打比方,代理则是中介,首先要明确的是:不论是静态代理还是动
态代理,被代理对象都要有一个实现的接口
2.1 静态代理
静态代理的本质是:用户调用目标对象的方法是通过代理对象调用的,其中代理对象中实现并组合了目标对象实现的接口,因此可以在代理对象中调用目标对象的方法。
1. 目标对象实现的接口
package cn.huawei.ProxyPattern;
public interface TargetInterface {
void method1();
}
2. 目标对象
package cn.huawei.ProxyPattern;
public class Target implements TargetInterface {
@Override
public void method1() {
System.out.println("STEP3: 睡觉");
}
}
3. 静态代理对象
package cn.huawei.ProxyPattern;
public class StaticProxy implements TargetInterface{
private TargetInterface proxied;
@Override
public void method1() {
System.out.println("STEP1: 洗脸");
System.out.println("STEP2: 刷牙");
proxied.method1();
}
public StaticProxy(TargetInterface proxied) {
this.proxied = proxied;
}
}
4.Main函数
package cn.huawei.ProxyPattern;
public class Main {
public static void fun1(TargetInterface iface){
iface.method1();
}
public static void main(String[] args) {
System.out.println("代理之前:");
fun1(new Target());
System.out.println("代理之后:");
fun1(new StaticProxy(new Target()));
}
}
5.运行结果
代理之前:
STEP3: 睡觉
==========
代理之后:
STEP1: 洗脸
STEP2: 刷牙
STEP3: 睡觉
2.2 动态代理
动态代理的本质和静态代理是不同的,不同点如下:
1. 静态代理需要用户写代理对象StaticTarget,而动态代理是JVM在内存中生成代理对象
2. 动态代理的代理对象是通过反射机制调用目标对象的方法,而静态代理是通过在代理对象中组合目标对象实现的接口来实现调用目标对象方法的功能
3. 动态代理的代理对象和目标对象实现了相同的接口
【参考资料】:Java中的反射机制
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy; public class Main {
public static void main(String[] args) {
FBB fbb = new FBB(); /*
* 参数讲解:
* 1. 被代理对象的类加载器
* 2. 被代理对象实现的接口的字节码数组 // 也可以是 new Class[] { MX.class }
* 3. InvocationHandler
* 4. 返回值要和接口的字节码数组中的接口类型一致
*/
MX proxy = (MX) Proxy.newProxyInstance(MX.class.getClassLoader(), fbb.getClass().getInterfaces(), new InvocationHandler() {
/*
* InvocationHandler可以理解为一组规范,代理对象proxy调用目标对象几次方法,该方法就执行几次
* invoke:执行代理对象的方法
* proxy:代理对象
* method:目标对象方法的字节码Method对象,来自FBB中的方法
* args:被代理对象中方法的参数,来自FBB方法中的参数
* @return 返回代理方法的结果
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
if ("sing".equals(method.getName())) { //针对唱歌项目,如果加钱可以唱歌曲2
System.out.println("加钱 歌曲2"); //增强方法
Object invoke = method.invoke(fbb, args);
return invoke; //返回sing()结果
} else {
return method.invoke(fbb, args);
}
}
});
System.out.println(proxy.sing());
System.out.println(proxy.dance());
}
} interface MX { //明星接口,默认明星都会唱歌和跳舞
String sing();
String dance();
} class FBB implements MX { //目标对象(被代理对象)
@Override
public String sing() { //FBB默认只会唱歌曲1
return "fbb sing 歌曲1";
} @Override
public String dance() {//FBB默认只会跳舞蹈1
return "fbb dance 舞蹈1 ";
}
}
执行结果
=======
加钱 歌曲2
fbb sing 歌曲1
fbb dance 舞蹈1
【design pattern】代理模式的更多相关文章
- Thinking In Design Pattern——MVP模式演绎
原文<Thinking In Design Pattern——MVP模式演绎>不知为何丢失了,故重新整理了一遍. 目录 What Is MVP Domain Model StubRepos ...
- proxy pattern 代理模式
常用的几种代理模式简要说明如下: (1) 远程代理(Remote Proxy):为一个位于不同的地址空间的对象提供一个本地的代理对象,这个不同的地址空间可以是在同一台主机中,也可是在另一台主机中,远 ...
- Proxy Design Pattern 代理设计模式
代理设计模式.此模式是用于serverclient排序.互联网接入,也经常使用的类代理,我觉得这种感觉很复杂.但是,这种设计模式本身是非常easy的. 是一类调用另一个类的功能.客户调用类,实际工作是 ...
- Design Pattern - 命令模式
一般执行一个操作的过程, 创建对象, 并调用对象的函数, 函数执行, 返回 比如下面的类图, client直接调用Receiver.action 而命令模式, 抽象出command对象, 并在comm ...
- Design Pattern - 访问者模式
访问者模式 访问者模式(Visitor), 表示一个作用于某对象结构中的各元素的操作.它使你可以在不改变各元素的类的前提下定义作用于这些元素的新操作. 这个模式相对比较复杂, 而又很少能被用上, 拿G ...
- js设计模式系列之(一)请节约你的请求-代理模式
What’s the proxy pattern? 代理模式其实就是将违反单一性原则的类给抽离出来,尽量满足开放和封闭的原则. 相当于一个类的行为只是一种,但是你可以给这个类添加额外的行为.比如: 一 ...
- 深入浅出设计模式——代理模式(Proxy Pattern)
模式动机在某些情况下,一个客户不想或者不能直接引用一个对象,此时可以通过一个称之为“代理”的第三者来实现间接引用.代理对象可以在客户端和目标对象之间起到中介的作用,并且可以通过代理对象去掉客户不能看到 ...
- NET设计模式 第二部分 结构性模式(13):代理模式(Proxy Pattern)
代理模式(Proxy Pattern) ——.NET设计模式系列之十四 Terrylee,2006年5月 摘要:在软件系统中,有些对象有时候由于跨越网络或者其他的障碍,而不能够或者不想直接访问另一个对 ...
- 代理模式(Proxy pattern)
代理模式(proxy pattern):作用:为其他对象提供一种代理,以控制对这个对象的访问.代理对象在客户端对象和目标对象之间起中介的作用. 代理模式涉及到的角色: 抽象角色:声明真实对象和代理对象 ...
随机推荐
- ES6躬行记(24)——代理和反射
代理和反射是ES6新增的两个特性,两者之间是协调合作的关系,它们的具体功能将在接下来的章节中分别讲解. 一.代理 ES6引入代理(Proxy)地目的是拦截对象的内置操作,注入自定义的逻辑,改变对象的默 ...
- 11.2NOIP模拟赛
/* 根右左遍历后最长上升子序列 */ #include<iostream> #include<cstdio> #include<cstring> #include ...
- 文件cp功能
#include<stdio.h> #include<unistd.h> #include<fcntl.h> #include<string.h> in ...
- 洛谷 P1966 火柴排队
题目描述 涵涵有两盒火柴,每盒装有 n 根火柴,每根火柴都有一个高度. 现在将每盒中的火柴各自排成一列, 同一列火柴的高度互不相同, 两列火柴之间的距离定义为:∑(ai−bi)2 其中ai 表示 ...
- 《windows核心编程系列》二十一谈谈基址重定位和模块绑定
每个DLL和可执行文件都有一个首选基地址.它表示该模块被映射到进程地址空间时最佳的内存地址.在构建可执行文件时,默认情况下链接器会将它的首选基地址设为0x400000.对于DLL来说,链接器会将它的首 ...
- CF1140G Double Tree
题解 首先如果我们要确定出每个\(dis_{i \to i+1 , i \in odd}\) 这个可以用两遍树形\(DP\)来解决 一遍是考虑走子树子树绕过来的 一遍是考虑从走祖先绕过来的 然后就可以 ...
- Lightoj 1090 - Trailing Zeroes (II)
题目连接: http://www.lightoj.com/volume_showproblem.php?problem=1090 题目大意: 给出n,r,p,q四个数字1<=n,r,p,q< ...
- bryce1010专题训练——线段树习题汇总
一.区间查询,无单点更新 hdu2795 Billboard Time Limit: 20000/8000 MS (Java/Others) Memory Limit: 32768/32768 ...
- clock()函数的返回值精度问题
clock()函数返回值为1毫秒,就是0.001秒.clock函数功 能: 返回处理器调用某个进程或函数所花费的时间.用 法: clock_t clock(void);说明:clock_t其实就是lo ...
- 贪心 UVALive 6834 Shopping
题目传送门 /* 题意:有n个商店排成一条直线,有一些商店有先后顺序,问从0出发走到n+1最少的步数 贪心:对于区间被覆盖的点只进行一次计算,还有那些要往回走的区间步数*2,再加上原来最少要走n+1步 ...