代理模式(Proxy)--静态代理
1,代理模式的概念
代理模式:为其他对象提供一种代理,以控制对这个对象的访问(代理对对象起到中介的作用,可去掉功能服务或者添加额外的服务)
2,代理模式的分类
(1)远程代理:类似于客户机服务器模式
为不同地址空间的对象提供局域网代表对象
举例说明:通过远程代理监控各个店铺

(2)虚拟代理:根据需要将资源消耗很大的对象进行延迟,真正需要的时候进行创建
举例说明:当对一个网页进行记载的时候,会先使用一张默认的图片代替要加载的图片,将文字等其他信息和默认的图片一起加载出来,在慢慢加载指定的图片。
(3)保护代理:控制对一个对象的访问权限
举例说明:在浏览某个网站的时候,未登录状态,只能浏览网站,只有在登陆后才可以发帖。
(4)智能引用代理:提供对目标对象的额外的服务
举例说明:火车票代购处,不仅提供了销售火车票的功能,还提供了电话预约的功能。
3,智能引用代理---静态代理:


模拟场景:车行驶的路上,代理添加新功能:记录行驶的时间
(1)在不使用代理的情况下:
S1:编写行驶接口类Moveable
package com.songyan.proxy;
public interface Moveable {
void move();
}
S2:编写小汽车类实现Moveable接口
并实现Moveable中定义的行驶方法(move())
在行驶方法中获取运行开始时,运行结束时的时间,做差获取运行的时间,由于程序运行太快,使用Thread的sleep()休眠1秒
package com.songyan.proxy;
import java.util.Random;
public class Car implements Moveable {
public void move() {
long startTime=System.currentTimeMillis();
System.out.println("汽车行驶的开始时间:"+startTime);
//实现开车
try {
Thread.sleep(new Random().nextInt(1000));
System.out.println("汽车行驶中");
} catch (InterruptedException e) {
e.printStackTrace();
}
long endTime=System.currentTimeMillis();
System.out.println("骑车行驶的终止时间:"+endTime);
System.out.println("汽车行驶时间共计:"+(endTime-startTime)+"毫秒~");
}
}
S3:编写测试类,是小汽车行驶
package com.songyan.proxy;
public class Client {
public static void main(String[] args) {
Car car= new Car();
car.move();
}
}
S4:运行结果

(2)代理实现记录汽车行驶的时间---继承
S1:新建Car2继承Car
S2:重写move方法
S3:将car中的记录时间的部分抽取出来,只留下汽车行驶的部分
package com.songyan.proxy;
/**
*
* @author sy
*通过继承Car2实现对Car的代理
*/
public class Car2 extends Car{
@Override
public void move() {
long startTime=System.currentTimeMillis();
System.out.println("开始行驶~~~~"); super.move(); long endTime=System.currentTimeMillis();
System.out.println("行驶结束~~~~~~~~~~");
System.out.println("汽车行驶时间:"+(endTime-startTime)+"毫秒~");
} }
package com.songyan.proxy;
import java.util.Random;
public class Car implements Moveable {
public void move() {
System.out.println("移动~~~~~~~~~~~~");
try {
Thread.sleep(new Random().nextInt(1000));
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
S4:编写测试类测试
package com.songyan.proxy;
public class Client {
public static void main(String[] args) {
/*Car car=new Car();
car.move();*/
Moveable car2=new Car2();
car2.move();
}
}
输出结果

(3)代理实现记录汽车行驶的时间---聚合
S1:编写Car3实现Moveable接口
S2:声明一个Car类型的变量
S3:并通过构造函数进行初始化
S4:重写move方法,调用car的move()在写逻辑代码。
package com.songyan.proxy;
/**
*
* @author sy
* 通过聚合的方式实现代理
*/
public class Car3 implements Moveable{
private Car car;
/**
* 聚合:一个类当中
*/
public void move() {
//计时开始
long startTime=System.currentTimeMillis();
System.out.println("开始行驶~~~~"); //调用car的move方法
car.move(); //计时结束
long endTime=System.currentTimeMillis();
System.out.println("行驶结束~~~~~~~~~~");
System.out.println("汽车行驶时间:"+(endTime-startTime)+"毫秒~");
} /**
* 通过构造函数将car传进来
* @param car
*/
Car3(Car car)
{
super();
this.car=car;
} }
思考:以上是两种实现代理的方式,哪一种更好呢?
功能拓展:上面的代理中知识拓展了记录行驶时间的功能,如果需要实现功能叠加,两种方式有该如何实现?
(1)继承的方式
先记录日志在记录行驶时间:需要编写Car3继承Car.
先记录行驶时间,在记录日志:又需要编写Car4继承Car
不难看出继承的方式代码的可重用性不高
(2)聚合的方式(可以很好的解决上述问题)
S1:编写实现记录时间的代理类
package com.songyan.proxy;
/**
* 记录时间的代理类
* @author sy
*
*/
public class CarTimeProxy implements Moveable
{
private Moveable m;
public void move() {
System.out.println("时间记录开始~~");
m.move();
System.out.println("时间记录结束~~"); }
public CarTimeProxy(Moveable m)
{
this.m=m;
}
}
S2:编写记录日志的代理类
package com.songyan.proxy;
/**
* 就日志的代理类
* @author sy
*
*/
public class CarLogProxy implements Moveable{
private Moveable m;
public void move() {
System.out.println("log开始~~~~");
m.move();
System.out.println("log结束~~~~~~");
}
public CarLogProxy(Moveable m)
{
this.m=m;
} }
S3:编写测试类(1)先记录日志在记录时间
package com.songyan.proxy;
/**
* 代理组合测试类
* @author sy
* 要求代理:先记录日志,在记录时间
*/
public class CarTest {
public static void main(String[] args) {
Moveable m=new Car();
Moveable time=new CarTimeProxy(m);
Moveable log=new CarLogProxy(time);
log.move();
}
}
运行结果:

S4:编写测试类2(先记录时间在记录日志)
不行需要重新写代理类
只需要在调用的时候调整顺序即可
package com.songyan.proxy;
/**
* 代理组合测试类
* @author sy
* 要求代理:先记录时间,在记录日志
*/
public class CarTest {
public static void main(String[] args) {
Moveable m=new Car();
Moveable log=new CarLogProxy(m);
Moveable time=new CarTimeProxy(log);
time.move();
}
}
输出结果:

由上可以看出:聚合要比继承更好用。
思考:我们上面实现的都是对Car的代理,如果我们下需要对火车,自行车实现相同的代理,我们又该如何实现呢?
这里就需要用到动态代理技术了:具体请点击此处查看另一篇博客。
代理模式(Proxy)--静态代理的更多相关文章
- Spring框架_代理模式(静态代理,动态代理,cglib代理)
共性问题: 1. 服务器启动报错,什么原因? * jar包缺少.jar包冲突 1) 先检查项目中是否缺少jar包引用 2) 服务器: 检查jar包有没有发布到服务器下: ...
- java 代理模式(静态代理、动态代理、Cglib代理) 转载
Java的三种代理模式 1.代理模式 代理(Proxy)是一种设计模式,提供了对目标对象另外的访问方式;即通过代理对象访问目标对象.这样做的好处是:可以在目标对象实现的基础上,增强额外的功能操作,即扩 ...
- Java三种代理模式:静态代理、动态代理和cglib代理
一.代理模式介绍 代理模式是一种设计模式,提供了对目标对象额外的访问方式,即通过代理对象访问目标对象,这样可以在不修改原目标对象的前提下,提供额外的功能操作,扩展目标对象的功能. 简言之,代理模式就是 ...
- java代理模式之静态代理
作为一个初级开发者,可能不会接触到代理模式,但是在很多框架的使用中都不知不觉使用了代理模式,比如servlet的过滤器链,spring的AOP,以及spring mvc的拦截器等.所以了解代理模式对于 ...
- 代理模式之静态代理,JDK动态代理和cglib动态代理
代理模式,顾名思义,就是通过代理去完成某些功能.比如,你需要购买火车票,不想跑那么远到火车站售票窗口买,可以去附近的火车票代售点买,或者到携程等第三方网站买.这个时候,我们就把火车站叫做目标对象或者委 ...
- java 编程基础 Class对象 反射:代理模式和静态代理
生活中的代理 类(对象)代理模式 代理模式是面向对象编程中比较常见的设计模式. 1. 用户只关心接口功能,而不在乎谁提供了功能.上图中接口是 Subject 2. 接口真正实现者是上图的 RealSu ...
- 【Java】代理模式,静态代理和动态代理(基于JDK或CGLib)
当我们需要在一个方法之前或之后添加一段逻辑时,自然会想到使用代理类.代理类帮我们代理了实际类的调用,然后可以在实际调用之前和之后添加一些逻辑,从而不浸入实际类. 拓展:由于代理类能在实际类调用之前和之 ...
- 结构型模式 -- 代理模式(静态代理&动态代理)
静态代理: 真实角色和代理角色实现相同的接口,代理角色拥有真实角色的引用.代理角色去执行方法,对于某些"真正"需要真实角色自己执行的方法时,在代理角色内部就调用真实角色的方法,其他 ...
- Java的三种代理模式:静态代理/JDK动态代理/Cglib动态代理
1.静态代理:需要定义接口或者父类,目标对象与代理对象均实现同一接口或继承同一父类. 2.JDK动态代理:需要目标对象实现一个接口,通过动态反射的机制,生成代理对象,实现同一个接口 3.Cglib动态 ...
- 《Proxy系列专题》:代理模式(静态、JDK、CGLib)
<Proxy系列专题>:代理模式(静态.JDK.CGLib)使用 现象:在如今互联网时代,项目的复杂度不断的提升,有些场景下需要一定的设计优化来支撑业务的扩展,如为了不改动原始类,但需要对 ...
随机推荐
- linux socket c/s上传文件
这是上传文件的一个示例,可以参照自行修改成下载或者其它功能. 在上传时,需要先将文件名传到服务器端,这是采用一个结构体,包含文件名及文件名长度(可以用于校验),防止文件名乱码. client #inc ...
- ls目录结构
命令ls ls -l = ll -l 详细信息-a 查看隐藏的文件或目录-d 只看目录本身,不列出目录下面的文件和目录 一起使用一般 ls -ld-t 以时间先后排序-i 显示文件节点-h 显示字节大 ...
- mac系统安装/升级node
一.安装 1.node 是通过brew来安装的,所以第一步先安装brew ruby -e "$(curl -fsSL https://raw.githubusercontent.com/Ho ...
- 第十四篇:JavaScript
本篇内容 简介 使用 DOM 一. 简介 JavaScript 是世界上最流行的编程语言. 这门语言可用于 HTML 和 web,更可广泛用于服务器.PC.笔记本电脑.平板电脑和智能手机等设备. Ja ...
- A公司 推荐算法大赛 总结
一.介绍 ♦通过用户前四个月(04.15~08.15)的用户行为预测用户第五个月(08.15~09.15)将会购买的品牌.用户共有四种行为(type)分别是:点击(0).购买(1).购物车(2).收藏 ...
- jquery defered的progress方法实现进度条
效果如图: 实现代码: <!DOCTYPE html> <html lang="en"> <head> <meta charset=&qu ...
- NOIP2017赛前考试注意事项总结
考前: 考试前把读入优化和库以及对拍文件打好做好准备工作,另外注意放松心态,太紧张了肯定考不好··将自己的注意力集中起来 考场策略: 考试的基本策略是对每于道题先想个20分钟,如果想不出个靠谱的方 ...
- 基于RRT的机器人自主探索建图
一.方法讲解: 本项目分为三个部分:机器人周围一定范围内基于RRT的全局检测, 根据上一步检测的未知区域点执行sklearn.cluster.MeanShift聚类,获取聚类中心: 根据聚类中心计算各 ...
- getprop 获取android系统属性
Android属性系统 property_get/property_set (很透彻)http://www.blogjava.net/MEYE/articles/359773.html getpro ...
- MFC 屏幕截图(libjpeg bmp转jpg)
项目中需要用到的功能. void Screenshot() { CDC *pDC; pDC = CDC::FromHandle(GetDC(GetDesktopWindow())); if(pDC = ...