在上篇博客中个。咱们一起组建了一个容器,里面封装了业务,这样,咱们就将业务和服务的组装放到了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的更多相关文章

  1. 最新咕咆+鲁班+图灵+享学+蚂蚁+硅谷+源码 Java架构师资料《Java架构师VIP课程》

    最新的Java架构师完整资料,完整视频+源码+文档. 每一套都是一百多个G的资料,无密. JAVA架构师全套课程 咕泡学院互联网架构师第一期 咕泡学院互联网架构师第二期 咕泡学院互联网架构师第三期 博 ...

  2. paip.java 架构师之路以及java高级技术

    paip.java 架构师之路以及java高级技术 1.    Annotation 设计模式... 概念满天飞.ORM,IOC,AOP. Validator lambda4j memcache. 对 ...

  3. JAVA架构师要求

    JAVA架构师要求专业素质要求:1.理解架构师的职责和架构设计的目标.原则及取舍:2.精通架构模式,Transaction.Security.Persistence等机制及实现,IOC.AOP.SOA ...

  4. 一位资深Java架构师的晋级心得

    架构师是什么? 是一个既需要掌控整体又需要洞悉局部瓶颈并依据具体的业务场景给出解决方案的团队领导型人物.一个架构师得需要足够的想像力,能把各种目标需求进行不同维度的扩展,为目标客户提供更为全面的需求清 ...

  5. JAVA架构师面试题 一

    基础题目 Java线程的状态 进程和线程的区别,进程间如何通讯,线程间如何通讯 HashMap的数据结构是什么?如何实现的.和HashTable,ConcurrentHashMap的区别 Cookie ...

  6. Java架构师之路 Spring学习笔记(一) Spring介绍

    前言 这是一篇原创的Spring学习笔记.主要记录我学习Spring4.0的过程.本人有四年的Java Web开发经验,最近在面试中遇到面试官总会问一些简单但我不会的Java问题,让我觉得有必要重新审 ...

  7. Java架构师必会的技能

    Java架构师必会的技能 我把它分为了五大专题 工程化专题 工程化专题 git git安装使用 git日常使用:fetch/pull/push/revert/rebase git分支管理git flo ...

  8. Java 架构师面试题

    基础题目 Java线程的状态 进程和线程的区别,进程间如何通讯,线程间如何通讯 HashMap的数据结构是什么?如何实现的.和HashTable,ConcurrentHashMap的区别 Cookie ...

  9. Java架构师必看,超详细的架构师知识点分享!

    在Java程序员行业中,有不少Java开发人员的理想是成为一名优秀的Java架构师,Java架构师的主要任务不是从事具体的软件程序的编写,而是从事更高层次的开发构架工作.他必须对开发技术非常了解,并且 ...

随机推荐

  1. JNI中java类型的简写

      在JNI中,当我们使用GetFieldID/GetStaticFieldID或GetMethodID/GetStaticMethodID及定义JNINativeMethod等时,我们需要表示成员变 ...

  2. Mysql优化ibdata1大小

    在MySQL数据库中,如果不指定innodb_file_per_table参数来单独存在每个表的数据,MySQL的数据都会存放在ibdata1文件.mysql ibdata1存放数据,索引等,是MYS ...

  3. 排序算法(Apex 语言)

    /* Code function : 冒泡排序算法 冒泡排序的优点:每进行一趟排序,就会少比较一次,因为每进行一趟排序都会找出一个较大值 时间复杂度:O(n*n) 空间复杂度:1 */ List< ...

  4. [BZOJ1935][SHOI2007]Tree 园丁的烦恼(离线+树状数组)

    题意 给出第一象限的n个点,有m次询问,每次询问一个矩形中的点的个数.(0<=n,m<=500000,0<=xi,yi<=10000000) 题解 一眼望去不可做. 用二位前缀 ...

  5. java源码学习

    Collection List ArrayList LinkedList Vector Stack Set HashSet TreeSet Map HashMap TreeMap LinkedHash ...

  6. java 线程传参 方式

    第一类:主动向线程传参   public class ThreadTest extends Thread { public ThreadTest() { } /** * 第一种通过构造方法来传递参数 ...

  7. 【J-meter】正则表达式提取

    当获取的值中含有折行,可采用下面的办法解决:

  8. caioj 1086 动态规划入门(非常规DP10:进攻策略)

    一开始看到题目感觉很难 然后看到题解感觉这题贼简单,我好像想复杂了 就算出每一行最少的资源(完全背包+二分)然后就枚举就好了. #include<cstdio> #include<a ...

  9. python通过sigar收集服务器信息

    http://blog.csdn.net/mirahs/article/details/49681787

  10. 洛谷 P3663 [USACO17FEB]Why Did the Cow Cross the Road III S

    P3663 [USACO17FEB]Why Did the Cow Cross the Road III S 题目描述 Why did the cow cross the road? Well, on ...