设计模式之jdk动态代理模式、责任链模式-java实现
设计模式之JDK动态代理模式、责任链模式
需求场景
- 当我们的代码中的类随着业务量的增大而不断增大仿佛没有尽头时,我们可以考虑使用动态代理设计模式,代理类的代码量被固定下来,不会随着业务量的增大而增大。
- 当我们想要在类方法调用前、调用后等自动插入一些额外的功能时,可以考虑使用动态代理来实现而不用修改类的源代码。
- 动态代理对象是动态生成的,不是定义好的,所以动态代理模式可以作为一种解除耦合的手段。
- 动态代理模式也可以作为责任链设计模式的实现手段。
前提条件
JDK动态代理的被代理对象是需要实现接口的。
成熟案例
Spring框架的AOP编程,就是使用动态代理实现的。
手写案例
首先,我们的被代理对象要实现一个公共的接口,这样可以做到一个大类型的整体代理,下面是这个接口及其实现类。
public interface IMyClass{
public void saySomething(String msg);
}
public class MyClassImpl implements IMyClass{
@Override
public void saySomething(String msg){
System.out.println(msg);
}
}
现在我们想要使用代理对象来代理这个类,但是代理对象从哪里来呢?所以,我们需要写一个动态产生代理对象的“工厂”。这个“工厂”需要实现一个叫做 InvocationHandler 的接口。
public class MyProxy<T> implements InvocationHandler {
//实际对象,即被代理对象
private T target;
//绑定方法,获取代理对象
public T bind(T target) {
this.target = target;
return (T) Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("调用方法前做的事");
//调用方法
Object result = method.invoke(target, args);
System.out.println("调用方法后做的事");
return result;
}
}
现在我们有了被代理对象,有了代理对象的生产“工厂”,就可以开始试一试了。
public class Test{
public static void main(String[] args){
//获取代理对象
IMyClass myClassProxy = new MyProxy<IMyClass>().bind(new MyClassImpl());
myClassProxy.saySomething("Hello World!");
}
}
结果如下:
调用方法前做的事
Hello World!
调用方法后做的事
模拟拦截器
使用jdk代理模式很容易实现一个拦截器,下面是我们的拦截器的接口和一个实现类。
public interface Interceptor{
public boolean before(Object proxy, Method method, Object[] args, Object target) throws Exception;
public Object around(Object proxy, Method method, Object[] args, Object target) throws Exception;
public void after(Object proxy, Method method, Object[] args, Object target) throws Exception;
}
public class InterceptorImpl implements Interceptor{
@Override
public boolean before(Object proxy, Method method, Object[] args, Object target) throws Exception{
System.out.println("Before!");
if(((String)args[0]).length() <= 10){
//未通过验证
System.out.println("It's too short!");
return false;
}
//通过验证
return true;
}
@Override
public Object around(Object proxy, Method method, Object[] args, Object target) throws Exception{
System.out.println("add something satrt!");
args[0] += " world!";
Object result = method.invoke(target, args);
System.out.println("add something end!");
return result;
}
@Override
public void after(Object proxy, Method method, Object[] args, Object target) throws Exception{
System.out.println("After!");
}
}
将我们的代理类生产“工厂”稍作修改
public class MyProxy<T> implements InvocationHandler {
//实际对象,即被代理对象
private T target;
//我们的拦截器类名
private String interceptorName;
//绑定方法,获取代理对象
public T bind(T target, String interceptorName) {
this.target = target;
this.interceptorName = interceptorName;
return (T) Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
if(null == this.interceptorName){
return method.invoke(target, args);
}
Object result = null;
Interceptor interceptor = (Interceptor)Class.forName(this.interceptorName).newInstance();
if(interceptor.before(proxy, method, args, target)){
result = method.invoke(target, args);
}else{
result = interceptor.around(proxy, method, args, target);
}
interceptor.after(proxy, method, args, target);
return result;
}
}
试一试我们的拦截器
public class Test {
public static void main(String[] args) {
IMyClass myClassProxy = new MyProxy<IMyClass>().bind(new MyClassImpl(), "com.test.InterceptorImpl");
myClassProxy.saySomething("hello");
System.out.println("***************************************");
myClassProxy.saySomething("123456789012");
}
}
结果如下:
Before!
It's too short!
add something satrt!
hello world!
add something end!
After!
Before!
123456789012
After!
可以看到,我们成功实现了拦截的功能,增强了方法。
进阶责任链模式
责任链模式可以基于Jdk动态代理实现,模拟一下多个拦截器对同一个对象的拦截,你可以想象成一个员工要请假,把请假条先交给小组长签字,然后给项目经理签字,再返回给小组长,再到自己手上的场景。这个被代理的对象就是这个请假条,这些拦截器就是小组长和项目经理。
首先,我们再实现一个项目经理,把“小组长”的代码稍稍改一下即可。
public class InterceptorImpl1 implements Interceptor{
@Override
public boolean before(Object proxy, Method method, Object[] args, Object target) throws Exception{
System.out.println("Before1!");
if(((String)args[0]).length() <= 10){
//未通过验证
System.out.println("It's too short1!");
return false;
}
//通过验证
return true;
}
@Override
public Object around(Object proxy, Method method, Object[] args, Object target) throws Exception{
System.out.println("add something satrt1!");
args[0] += " world1!";
Object result = method.invoke(target, args);
System.out.println("add something end1!");
return result;
}
@Override
public void after(Object proxy, Method method, Object[] args, Object target) throws Exception{
System.out.println("After1!");
}
}
下面开始“递交请假条”。
public class Test {
public static void main(String[] args) {
IMyClass myClassProxy = new MyProxy<IMyClass>().bind(new MyClassImpl(), "com.test.InterceptorImpl");
IMyClass myClassProxy1 = new MyProxy<IMyClass>().bind(myClassProxy, "com.test.InterceptorImpl1");
myClassProxy1.saySomething("hello");
System.out.println("***************************************");
myClassProxy1.saySomething("123456789012");
}
}
结果如下
Before1!
It's too short1!
add something satrt1!
Before!
hello world1!
After!
add something end1!
After1!
Before1!
Before!
123456789012
After!
After1!
看,是不是外面又套了一层?这个想不想是Spring的AOP编程的效果呢?其实原理就是这样!
总结
动态代理模式是一种非常重要的设计模式。我们可以看到,很多的框架都大量使用动态代理模式来实现其特殊的功能,这个设计模式需要好好好掌握。
责任链设计模式可以看作是一种扩展,仔细体会一下就能够领会其中的奥妙。大家可以仔细看看责任链模式案例运行结果的顺序,有助于理解哦!
开发模式是前任对某一类相似问题的解决方案进行的总结,我们可以从中吸取经验,但也不能盲目使用,也不能为了使用设计模式而使用设计模式。作为一个coder,要始终保持自己活跃的思维,不能将思维局限在一个设计模式上。开发熟练度在一定程度上决定了你的工作效率,但是思维的活跃度和发散性决定了你的天花板,也在很大程度上决定了你的工作效率。
你虽然在为别人工作,但是,你也在创造自己的世界!
设计模式之jdk动态代理模式、责任链模式-java实现的更多相关文章
- Java 设计模式系列(二二)责任链模式
Java 设计模式系列(二二)责任链模式 责任链模式是一种对象的行为模式.在责任链模式里,很多对象由每一个对象对其下家的引用而连接起来形成一条链.请求在这个链上传递,直到链上的某一个对象决定处理此请求 ...
- 设计模式之Jdk动态代理
什么是动态代理呢?就是在java的运行过程中,动态的生成的代理类.(为了更熟悉的了解动态代理,你必须先熟悉代理模式,可点击设计模式之代理模式 阅读)我们知道java属于解释型语言,是在运行过程中,寻找 ...
- 设计模式之(十四)责任链模式(Chain of Responsibility)
在业务场景中,有很多是需要审批的.审核方式还可能常常发生变化,而责任链模式就是为了解决这种场景的情况的. 责任链模式定义:十多个对象都有机会处理请求,从而避免发送者和接受者之间的耦合关系.讲这些对象连 ...
- 《Android源码设计模式》--状态模式--责任链模式--解释器模式--命令模式--观察者模式--备忘录模式--迭代器模式
[状态模式] No1: Wifi设置界面是一个叫做WifiSetting的Fragment实现的 No2: 在不同的状态下对于扫描Wifi这个请求的处理是完全不一样的.在初始状态下扫描请求被直接忽略, ...
- Java设计模式之JDK动态代理原理
动态代理核心源码实现public Object getProxy() { //jdk 动态代理的使用方式 return Proxy.newProxyInstance( this.getClass(). ...
- [19/04/29-星期一] GOF23_行为型模式(责任链模式、迭代器模式)
一.行为模式 二.责任链模式(chainOfResponsibility) [一个请求类] /*** 请假类 * 封装请假的基本信息 */ package cn.sxt.chainOfResponsi ...
- Java设计模式从精通到入门一 责任链模式
一直都想对设计模式有一个深刻的认识,这样对于阅读源码的时候就不会那么吃力了.于是有了想要记录下设计模式的笔记.打算从自己不怎么熟悉的设计模式开始写,里面穿插着一点自己的想法,希望自己写完后,会又一 ...
- 设计模式学习总结(十)责任链模式(Chain Of Responsibility)
责任链主要指通过一连串的操作来实现某项功能或者在处理相关的业务时,对于自己的业务则进行处理,反之,对于不属于自己的业务,则进行下发! 一.示例展示: 以下例子主要通过对煤矿对井下警告信息的处理来进 ...
- 《精通Python设计模式》学习行为型之责任链模式
感觉是全新的学习了. 因为在以前的工作中,并没有有意识的去运用哪一种编程模式. 以后要注意的了. 这才是高手之路呀~ class Event: def __init__(self, name): se ...
随机推荐
- vs2015 iis express启动不了及安装DotNetCore.1.0.0-VS2015Tools.Preview2失败的解决方法
直接用管理员账户打开cmd,进入exe所在的文件夹在运行命令DotNetCore.1.0.0-VS2015Tools.Preview2.exe SKIP_VSU_CHECK=1不要加引号. PS:如果 ...
- 微信小程序tab栏切换
Wxml代码:<view class="body"> <view class="nav bc_white"> <view clas ...
- 实现A-Z滑动检索菜单
我们实现一个A-Z滑动检索菜单,这是一个移动端非常常见的功能,页面效果图如下 在实现代码之前我们先了解下JS滚动事件和滑动事件 scrollTop 一个元素的scrollTop是这个元素的顶部 到 可 ...
- expdp/impdp数据泵分区表导入太慢了。添加不检查元数据参数提高效率:ACCESS_METHOD=DIRECT_PATH
分区表数据泵导入太慢,达不到客户的迁移要求导出语句如下:(10G单节点)userid='/ as sysdba'directory=milk_dirdumpfile=mon_%U.dmplogfile ...
- oo 第一次博客作业
oo 第一次博客作业 早在大一就听说了oo的各种传奇故事,大二下学期终于也开始了我的oo之旅. 基于度量来分析自己的程序结构 第一次作业 类图分析 耦合度分析 可以看出在第一次作业中,我的耦合度非常高 ...
- 【C语言基础】什么是字节?
字节就是存储数据的单位,并且是硬件所能访问的最小单位. 一个字节控制8位 int 类型为4个字节 long 类型为8个字节 char 类型为1个字节:Java中为两个字节
- Rsync备份功能总结
备份服务笔记====================================================================== Rsync是一款开源的.快速的.多功能的.可实 ...
- javaScript:压缩图片并上传
html代码: <input id="file" type="file" name="filesName"> js代码: var ...
- 6993: Dominoes(纯bfs)
题目描述Orz likes to play dominoes. Now giving an n*m chessboard and k dominoes whose size are 1*2, Orz ...
- ActiveMq Windows 配置优化
最近ActiveMQ 5.15.3 会报 OutofMemory的错误 在 wrapper.conf中 #wrapper.java.additional.8=-Dorg.apache.activemq ...