设计模式之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 ...
随机推荐
- Redis Sentinel实现高可用配置
一般情况下yum安装redis的启动目录在:”/usr/sbin” :配置目录在”/etc/redis/”在其目录下会有默认的redis.conf和redis-sentinel.conf redis高 ...
- (Python基础)文件操作
对文件操作流程 打开文件,得到文件句柄并赋值给一个变量 通过句柄对文件进行操作 关闭文件 现有文件如下 命名为7 years Once I was seven years old my momma t ...
- 获取物理内存total值和used值
1.使用 free -m 查看 2.物理内存total值 # free -m | grep Mem | awk '{print $2}' 3.物理内存used值 # free -m | grep Me ...
- 简单封装Redis做缓存
基于Redis封装一个简单的Python缓存模块 0. Docker Redis安装 参考: Get Docker CE for CentOS Docker 安装 Redis 安装Docker时错误s ...
- chown nagios:nagios -R /var/lib/php/
chown nagios:nagios -R /var/lib/php/
- loadrunner-关联
1.什么是关联? 将服务器动态返回的值,保存成一个参数,供后面需要使用到 此参数的地方使用 PS:1.一定是服务器返回的 2.动态变化的,静态不需要关联 3.服务器需要校验的 4.保存成一个参数 5. ...
- Java后端面试的一切技巧和常见的问题经验总结
原文地址:cnblogs.com/JavaArchitect/p/10011253.html 上周,密集面试了若干位Java后端候选人,工作经验在3到5年间.我的标准其实不复杂(适用90%小小小公司, ...
- pytorch 生成随机数
在使用PyTorch做实验时经常会用到生成随机数Tensor的方法,比如: torch.rand() torch.randn() torch.normal() torch.linespace() 在很 ...
- VUE 引入阿里图标库
本身项目是VUE, Element-UI项目, 所以内置Element-UI图标库 地址如下, https://element.eleme.cn/#/zh-CN/component/icon 使用时: ...
- JavaScript值全等判断
作为开发员,很多时候拿到数据之后都是要做数据判断,比较特别的情况就是我们需要做数组判断和对象判断,经常的我们就array === array ,object === object;但是可惜是我们得到的 ...