java架构解密——实时动态aop
在上篇博客中个。咱们一起组建了一个容器,里面封装了业务,这样,咱们就将业务和服务的组装放到了client,而client就相当于咱们的开发中使用到的配置文件。大家发现问题了吗?就是我不能动态修改了?业务容器在代理完毕后,又一次加入一个容器。这时候。是不被同意的,那这个容器就是个编译时起作用的容器,他就丧失了非常多的灵活性。
那怎么办呢?实际就是调换一下调用顺序就好。大家看看结果:
1,类图对照
改造后类图
2,代码实现:
2.1代理类的变化:
/**
* 打招呼动态代理类,给业务类加入功能
* 前一版本号为JDK代理实现
* 本次加入运行方法之前打印到控制台‘befor’
* 本次加入运行方法之后打印到控制台‘after’
*本次版本号为DGLIB代理
* 换代理类原因,JDK代理要求被代理类必须实现某接口,由于它底层实现是新建一个类,实现和被代理类同样的接口
* 用代理类新建的业务类代替原业务类
* CGLIB代理是新建一个类。继承自被代理类,用新建的代理类替换掉原业务类,就不须要接口了
*
*5.0版本号添加服务组装容器,将服务从代理类中抽离出去了,我们的代理类成为了一个bean
*6.0将服务容器定义为接口
*7.0添加业务容器
*8.0 业务容器不灵活,将获代替理放置到业务容器端,实现动态添加元素,也能拿到代理
* @author 许恕
* @version 3.0.0 , 2015年6月16日 下午3:20:13
*/
public class CGLibDynamicProxy implements MethodInterceptor {
//服务类容器
private IProxyMehds proxyMehds;
//代理工厂类:单例模式,优化内存开销
private static CGLibDynamicProxy instance = new CGLibDynamicProxy();
//构造函数
private CGLibDynamicProxy() {
}
//获取cglib代理工厂类
public static CGLibDynamicProxy getInstance() {
return instance;
}
/**
* 使用代理工厂生成某个类的代理
*
* @param cls 要代理的类
* @return 返回已经代理好的类
*/
@SuppressWarnings("unchecked")
public <T> T getProxy(Class<T> cls) {
return (T) Enhancer.create(cls, this);
}
//重写被代理对象的方法运行
//全部的方法运行。到反射的级别都是invoke,重写了这种方法,就重写了全部的方法运行,实现了代理
@Override
public Object intercept(Object target, Method method, Object[] args, MethodProxy proxy) throws Throwable {
//重要改进:从服务容器中运行方法,不再是写死的!
proxyMehds.beforeBean();
//方法正常运行的语句
Object result = proxy.invokeSuper(target, args);
//重要改进:从服务容器中运行方法。不再是写死的!
proxyMehds.afterBean();
return result;
}
//服务容器的get方法
public IProxyMehds getProxyMehds() {
return proxyMehds;
}
//服务容器的set方法
public void setProxyMehds(IProxyMehds proxyMehds) {
this.proxyMehds = proxyMehds;
}
}
2.2业务容器
/**
* 业务类容器:
* 用map盛放要切入服务的业务类
*
* @author 许恕
* @version 3.0.0 , 2015-6-29 14:25:21
*/
public class DoMehds implements IDoMehds {
//盛放运行业务的map
private HashMap<String,Object> DoBeans;
//代理类
private CGLibDynamicProxy proxy ;
/**
* 封装获取元素的方法。每次获取元素都封装一个代理
*
* @param 元素的名字
* @return 代理好的元素
*/
public Object getBean(String beanName){
return proxy.getProxy(DoBeans.get(beanName).getClass());
}
//获取业务map
public HashMap<String, Object> getDoBeans() {
return DoBeans;
}
//设置业务map
public void setDoBeans(HashMap<String, Object> doBeans) {
DoBeans = doBeans;
}
//获代替理类
public CGLibDynamicProxy getProxy() {
return proxy;
}
//设置代理类
public void setProxy(CGLibDynamicProxy proxy) {
this.proxy = proxy;
}
}
2.3 client
/**
* 运行代理程序的client
*
* @author 许恕
* @version 3.0.0 , 2015年6月16日 下午3:18:42
*/
public class Client {
//client运行的主方法:系统规定
public static void main(String[] args) {
//盛放方法运行前的对象的容器
HashMap<String,Object> beforBeans;
HashMap<String,Object> afterBeans;
//配制方法运行前要运行哪些方法
HashMap<String,String> beforMethods;
HashMap<String,String> afterMethods;
//实例以上定义的服务类map
beforMethods=new HashMap();
beforBeans=new HashMap();
afterMethods=new HashMap();
afterBeans=new HashMap();
//将服务类AspectClass1加入到方法运行要载入的服务类
beforBeans.put("AspectClass1", new AspectClass1());
beforBeans.put("AspectClass2", new AspectClass2());
afterBeans.put("AspectClass3", new AspectClass3());
afterBeans.put("AspectClass4", new AspectClass4());
//规定每一个服务类要运行的方法
beforMethods.put("AspectClass1", "SayHello");
beforMethods.put("AspectClass2", "SayGoodBye");
afterMethods.put("AspectClass3", "SayHi");
afterMethods.put("AspectClass4", "Eat");
//实例服务类容器。将四个map传入容器中
ProxyMehds proxyMehds =new ProxyMehds();
proxyMehds.setBeforBeans(beforBeans);
proxyMehds.setBeforMethods(beforMethods);
proxyMehds.setAfterBeans(afterBeans);
proxyMehds.setAfterMethods(afterMethods);
//实例业务map
HashMap<String,Object> DoBeansMap = new HashMap<String,Object> ();
//将业务装载到map中
DoBeansMap.put("dobeans1", new GreetingImpl());
DoBeansMap.put("dobeans2", new EatClass());
//将业务map装载到容器中
DoMehds doMehds = new DoMehds();
doMehds.setDoBeans(DoBeansMap);
//实例代理类
CGLibDynamicProxy cglib =CGLibDynamicProxy.getInstance();
//接受切面
cglib.setProxyMehds(proxyMehds);
doMehds.setProxy(cglib);
//接受要代理的对象
Greeting greeting = (Greeting)doMehds.getBean("dobeans1");
EatClass eatObject = (EatClass)doMehds.getBean("dobeans2");
//运行对象的某个方法
greeting.sayHello("Jack");
eatObject.Eat();
}
}
3。总结
代码的灵活,就像生活一般。在代码灵活的过程中。我们就会对照下生活,生活中。我们每一个人,事。物都是独立的对象。而在生活中的时间轴中。我们也是过程化的组织,我们的代码和生活。在思想上都是同样的。优化的不不过代码,还有思想啊。
java架构解密——实时动态aop的更多相关文章
- 最新咕咆+鲁班+图灵+享学+蚂蚁+硅谷+源码 Java架构师资料《Java架构师VIP课程》
最新的Java架构师完整资料,完整视频+源码+文档. 每一套都是一百多个G的资料,无密. JAVA架构师全套课程 咕泡学院互联网架构师第一期 咕泡学院互联网架构师第二期 咕泡学院互联网架构师第三期 博 ...
- paip.java 架构师之路以及java高级技术
paip.java 架构师之路以及java高级技术 1. Annotation 设计模式... 概念满天飞.ORM,IOC,AOP. Validator lambda4j memcache. 对 ...
- JAVA架构师要求
JAVA架构师要求专业素质要求:1.理解架构师的职责和架构设计的目标.原则及取舍:2.精通架构模式,Transaction.Security.Persistence等机制及实现,IOC.AOP.SOA ...
- 一位资深Java架构师的晋级心得
架构师是什么? 是一个既需要掌控整体又需要洞悉局部瓶颈并依据具体的业务场景给出解决方案的团队领导型人物.一个架构师得需要足够的想像力,能把各种目标需求进行不同维度的扩展,为目标客户提供更为全面的需求清 ...
- JAVA架构师面试题 一
基础题目 Java线程的状态 进程和线程的区别,进程间如何通讯,线程间如何通讯 HashMap的数据结构是什么?如何实现的.和HashTable,ConcurrentHashMap的区别 Cookie ...
- Java架构师之路 Spring学习笔记(一) Spring介绍
前言 这是一篇原创的Spring学习笔记.主要记录我学习Spring4.0的过程.本人有四年的Java Web开发经验,最近在面试中遇到面试官总会问一些简单但我不会的Java问题,让我觉得有必要重新审 ...
- Java架构师必会的技能
Java架构师必会的技能 我把它分为了五大专题 工程化专题 工程化专题 git git安装使用 git日常使用:fetch/pull/push/revert/rebase git分支管理git flo ...
- Java 架构师面试题
基础题目 Java线程的状态 进程和线程的区别,进程间如何通讯,线程间如何通讯 HashMap的数据结构是什么?如何实现的.和HashTable,ConcurrentHashMap的区别 Cookie ...
- Java架构师必看,超详细的架构师知识点分享!
在Java程序员行业中,有不少Java开发人员的理想是成为一名优秀的Java架构师,Java架构师的主要任务不是从事具体的软件程序的编写,而是从事更高层次的开发构架工作.他必须对开发技术非常了解,并且 ...
随机推荐
- BZOJ4259: 残缺的字符串 & BZOJ4503: 两个串
[传送门:BZOJ4259&BZOJ4503] 简要题意: 给出两个字符串,第一个串长度为m,第二个串长度为n,字符串中如果有*字符,则代表当前位置可以匹配任何字符 求出第一个字符串在第二个字 ...
- bzoj1009: [HNOI2008]GT考试(kmp+矩阵乘法)
1009: [HNOI2008]GT考试 题目:传送门 题解: 看这第一眼是不是瞬间想起组合数学??? 没错...这样想你就GG了! 其实这是一道稍有隐藏的矩阵乘法,好题! 首先我们可以简化一下题意: ...
- 解读邮箱正则表达式:^\w+([-+.]\w+)*@\w+([-.]\w+)*\.\w+([-.]\w+)*$
转自:http://www.cnblogs.com/joyceTING/archive/2013/05/09/3069089.html正则表达式 \w+([-+.]\w+)*@\w+([-.]\w+) ...
- Liunx-php7安装swoole扩展
Liunx-php7安装swoole扩展 标签(空格分隔): php 下载包 https://github.com/swoole/swoole-src/releases 安装过程 直接wget也行直接 ...
- 归档备份被删,GoldenGate无法抽取数据
发生错误如下,源端EXTRACT进程异常中止,查看日志,发现如下错误. 2014-07-23 01:32:13 ERROR OGG-00446 Oracle GoldenGate Captur ...
- PyCharm 2017 Mac 免注册版破解安装说明
PyCharm 2017 Mac 免注册版破解安装说明 下载完成安装包后,双击打开,将左侧拖拽至右侧应用程序,默认安装. 打开软件,在License server address中填入[http:// ...
- dstat---统计磁盘,CPU,IO,等相关信息
dstat命令是一个用来替换vmstat.iostat.netstat.nfsstat和ifstat这些命令的工具,是一个全能系统信息统计工具.与sysstat相比,dstat拥有一个彩色的界面,在手 ...
- 在kettle中实现数据验证和检查
在kettle中实现数据验证和检查 在ETL项目,输入数据通常不能保证一致性.在kettle中有一些步骤能够实现数据验证或检查.验证步骤能够在一些计算的基础上验证行货字段:过滤步骤实现数据过滤:jav ...
- javascript对象如何使用
javascript对象如何使用 一.总结 一句话总结:JavaScript 中的所有事物都是对象:字符串.数值.数组.函数... 因为函数是对象,所以自定义对象的创建中有种方法就是函数 1.js中的 ...
- google浏览器修改网页字符编码
google浏览器修改网页字符编码 直接在google浏览器的应用拓展程序里面搜 Charset,第一个就是 于是就有了