以下内容为原创,转载时请注明链接地址:http://www.cnblogs.com/tiantianbyconan/p/3336627.html

AOP(面向方面)的思想,就是把项目共同的那部分功能分离开来,比如日志记录,避免在业务逻辑里面夹杂着跟业务逻辑无关的代码。

下面是一个AOP实现的简单例子:

首先定义一些业务方法:

  1. /**
  2. * Created with IntelliJ IDEA.
  3. * Author: wangjie email:tiantian.china.2@gmail.com
  4. * Date: 13-9-23
  5. * Time: 下午3:49
  6. */
  7. public interface BussinessService {
  8. public String login(String username, String password);
  9. public String find();
  10. }
  11.  
  12. public class BussinessServiceImpl implements BussinessService {
  13. private Logger logger = Logger.getLogger(this.getClass().getSimpleName());
  14.  
  15. @Override
  16. public String login(String username, String password) {
  17. return "login success";
  18. }
  19.  
  20. @Override
  21. public String find() {
  22. return "find success";
  23. }
  24.  
  25. }
  1. /**
  2. * Created with IntelliJ IDEA.
  3. * Author: wangjie email:tiantian.china.2@gmail.com
  4. * Date: 13-9-24
  5. * Time: 上午10:27
  6. */
  7. public interface WorkService {
  8. public String work();
  9. public String sleep();
  10. }
  11.  
  12. public class WorkServiceImpl implements WorkService{
  13. @Override
  14. public String work() {
  15. return "work success";
  16. }
  17.  
  18. @Override
  19. public String sleep() {
  20. return "sleep success";
  21. }
  22. }

实现InvocationHandler接口,使用map来存储不同的InvocationHandler对象,避免生成过多。

  1. package com.wangjie.aoptest2.invohandler;
  2.  
  3. import java.lang.reflect.InvocationHandler;
  4. import java.lang.reflect.Method;
  5. import java.lang.reflect.Proxy;
  6. import java.util.Arrays;
  7. import java.util.HashMap;
  8. import java.util.logging.Logger;
  9.  
  10. /**
  11. * Created with IntelliJ IDEA.
  12. * Author: wangjie email:tiantian.china.2@gmail.com
  13. * Date: 13-9-23
  14. * Time: 下午3:47
  15. */
  16. public class LogInvoHandler implements InvocationHandler{
  17. private Logger logger = Logger.getLogger(this.getClass().getSimpleName());
  18.  
  19. private Object target; // 代理目标
  20. private Object proxy; // 代理对象
  21.  
  22. private static HashMap<Class<?>, LogInvoHandler> invoHandlers = new HashMap<Class<?>, LogInvoHandler>();
  23.  
  24. private LogInvoHandler() {
  25. }
  26.  
  27. /**
  28. * 通过Class来生成动态代理对象Proxy
  29. * @param clazz
  30. * @return
  31. */
  32. public synchronized static<T> T getProxyInstance(Class<T> clazz){
  33. LogInvoHandler invoHandler = invoHandlers.get(clazz);
  34.  
  35. if(null == invoHandler){
  36. invoHandler = new LogInvoHandler();
  37. try {
  38. T tar = clazz.newInstance();
  39. invoHandler.setTarget(tar);
  40. invoHandler.setProxy(Proxy.newProxyInstance(tar.getClass().getClassLoader(),
  41. tar.getClass().getInterfaces(), invoHandler));
  42. } catch (Exception e) {
  43. e.printStackTrace();
  44. }
  45. invoHandlers.put(clazz, invoHandler);
  46.  
  47. }
  48.  
  49. return (T)invoHandler.getProxy();
  50. }
  51.  
  52. @Override
  53. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  54.  
  55. Object result = method.invoke(target, args); // 执行业务处理
  56.  
  57. // 打印日志
  58. logger.info("____invoke method: " + method.getName()
  59. + "; args: " + (null == args ? "null" : Arrays.asList(args).toString())
  60. + "; return: " + result);
  61.  
  62. return result;
  63. }
  64.  
  65. public Object getTarget() {
  66. return target;
  67. }
  68.  
  69. public void setTarget(Object target) {
  70. this.target = target;
  71. }
  72.  
  73. public Object getProxy() {
  74. return proxy;
  75. }
  76.  
  77. public void setProxy(Object proxy) {
  78. this.proxy = proxy;
  79. }
  80. }

然后编写一个Test类测试:

  1. /**
  2. * Created with IntelliJ IDEA.
  3. * Author: wangjie email:tiantian.china.2@gmail.com
  4. * Date: 13-9-24
  5. * Time: 上午9:54
  6. */
  7. public class Test {
  8. public static Logger logger = Logger.getLogger(Test.class.getSimpleName());
  9. public static void main(String[] args) {
  10.  
  11. BussinessService bs = LogInvoHandler.getProxyInstance(BussinessServiceImpl.class);
  12. bs.login("zhangsan", "123456");
  13. bs.find();
  14.  
  15. logger.info("--------------------------------------");
  16.  
  17. WorkService ws = LogInvoHandler.getProxyInstance(WorkServiceImpl.class);
  18. ws.work();
  19. ws.sleep();
  20.  
  21. logger.info("--------------------------------------");
  22.  
  23. BussinessService bss = LogInvoHandler.getProxyInstance(BussinessServiceImpl.class);
  24. bss.login("lisi", "654321");
  25. bss.find();
  26.  
  27. }
  28. }

以后需要添加新的业务逻辑XXXService,只需要调用

XXXService xs = LogInvoHandler.getProxyInstance(XXXServiceImpl.class);

即可。

也可以模仿Spring等框架的配置,把bean的类名配置在xml文件中,如:

<bean id="bussinessService" class="com.wangjie.aoptest2.service.impl.BussinessServiceImpl">

然后在java代码中解析xml,通过Class.forName("com.wangjie.aoptest2.service.impl.BussinessServiceImpl");获得Class对象

然后通过LogInvoHandler.getProxyInstance(Class.forName("com.wangjie.aoptest2.service.impl.BussinessServiceImpl"));获得代理对象Proxy

再使用反射去调用代理对象的方法。

运行结果如下:

九月 24, 2013 11:08:03 上午 com.wangjie.aoptest2.invohandler.LogInvoHandler invoke
INFO: ____invoke method: login; args: [zhangsan, 123456]; return: login success
九月 24, 2013 11:08:03 上午 com.wangjie.aoptest2.invohandler.LogInvoHandler invoke
INFO: ____invoke method: find; args: null; return: find success
九月 24, 2013 11:08:03 上午 com.wangjie.aoptest2.Test main
INFO: --------------------------------------
九月 24, 2013 11:08:03 上午 com.wangjie.aoptest2.invohandler.LogInvoHandler invoke
INFO: ____invoke method: work; args: null; return: work success
九月 24, 2013 11:08:03 上午 com.wangjie.aoptest2.invohandler.LogInvoHandler invoke
INFO: ____invoke method: sleep; args: null; return: sleep success
九月 24, 2013 11:08:03 上午 com.wangjie.aoptest2.Test main
INFO: --------------------------------------
九月 24, 2013 11:08:03 上午 com.wangjie.aoptest2.invohandler.LogInvoHandler invoke
INFO: ____invoke method: login; args: [lisi, 654321]; return: login success
九月 24, 2013 11:08:03 上午 com.wangjie.aoptest2.invohandler.LogInvoHandler invoke
INFO: ____invoke method: find; args: null; return: find success

java使用动态代理来实现AOP(日志记录)的更多相关文章

  1. java动态代理与老式AOP实现

    JAVA的动态代理 代理模式是常用的java设计模式,他的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息.过滤消息.把消息转发给委托类,以及事后处理消息等.代理类与委托类之间通常会 ...

  2. java:struts框架2(方法的动态和静态调用,获取Servlet API三种方式(推荐IOC(控制反转)),拦截器,静态代理和动态代理(Spring AOP))

    1.方法的静态和动态调用: struts.xml: <?xml version="1.0" encoding="UTF-8"?> <!DOCT ...

  3. java --- 设计模式 --- 动态代理

    Java设计模式——动态代理 java提供了动态代理的对象,本文主要探究它的实现, 动态代理是AOP(面向切面编程, Aspect Oriented Programming)的基础实现方式, 动态代理 ...

  4. Java:动态代理小记

    Java:动态代理小记 对 Java 中的 动态代理,做一个微不足道的小小小小记 概述 动态代理:当想要给实现了某个接口的类中的方法,加一些额外的处理.比如说加日志,加事务等.可以给这个类创建一个代理 ...

  5. java的动态代理机制详解

    在学习Spring的时候,我们知道Spring主要有两大思想,一个是IoC,另一个就是AOP,对于IoC,依赖注入就不用多说了,而对于Spring的核心AOP来说,我们不但要知道怎么通过AOP来满足的 ...

  6. java中动态代理实现机制

    前言: 代理模式是常用的java设计模式,它的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息.过滤消息.把消息转发给委托类,以及事后处理消息等.代理类与委托类之间通常会存在关联关系 ...

  7. Java的动态代理机制详解(转)

    在学习Spring的时候,我们知道Spring主要有两大思想,一个是IoC,另一个就是AOP,对于IoC,依赖注入就不用多说了,而对于Spring的核心AOP来说,我们不但要知道怎么通过AOP来满足的 ...

  8. (转)java的动态代理机制详解

    原文出自:http://www.cnblogs.com/xiaoluo501395377/p/3383130.html 在学习Spring的时候,我们知道Spring主要有两大思想,一个是IoC,另一 ...

  9. [转载] java的动态代理机制详解

    转载自http://www.cnblogs.com/xiaoluo501395377/p/3383130.html 代理模式 代理模式是常用的java设计模式,他的特征是代理类与委托类有同样的接口,代 ...

随机推荐

  1. js组件在线编辑器插件、图表库插件、文件树插件

    在线编辑器插件: 一.kindeditor 二.UEditor 图表库插件: 一.echart 二.highchart 文件树插件: 一.zTree -- jQuery 树插件 http://www. ...

  2. ssl证书生成:cer&jks文件生成摘录

    一.生成.jks文件 1.keystore的生成: 分阶段生成:     keytool -genkey -alias yushan(别名) -keypass yushan(别名密码) -keyalg ...

  3. java类的初始化和对象的创建顺序

    学习java编程思想--类的初始化p146 类的加载顺序* 1加载器启动找到 xxx.class文件,通过extends关键字寻找基类,先加载基类* 2类初始化先初始化static成员变量和stati ...

  4. noip 模拟赛 匹配 //贪婪策略

    匹配(match.pas/match.c/match.cpp) [题目描述] 到了新的学期,Mcx痛苦的发现通用技术课居然是有实验课的,这样的话他就不得不放弃写作业的想法而去做一件类似于搭积木的事情. ...

  5. 【转】 Key/Value之王Memcached初探:三、Memcached解决Session的分布式存储场景的应用

    一.高可用的Session服务器场景简介 1.1 应用服务器的无状态特性 应用层服务器(这里一般指Web服务器)处理网站应用的业务逻辑,应用的一个最显著的特点是:应用的无状态性. PS:提到无状态特性 ...

  6. 15天玩转redis —— 第六篇 有序集合类型

    今天我们说一下Redis中最后一个数据类型 “有序集合类型”,回首之前学过的几个数据结构,不知道你会不会由衷感叹,开源的世界真好,写这 些代码的好心人真的要一生平安哈,不管我们想没想的到的东西,在这个 ...

  7. jdk1.6与Myeclipse的冲突造成的

    出现这样的错误时:ERROR:JDWP   Unable   to   get   JNI   1.2   environment   ,jvm-> GetEvn()   return   =- ...

  8. Oracle数据库,模糊查询、去重查询

    分组去重查询,并执行某一个函数 :select  分组字段,聚合函数 from 表名 where 条件 group by分组字段 select 分组字段,聚合函数 from 表名 where 条件 g ...

  9. jvm运行数据分布

    本人看的深入理解jvm(该版本的java se7) java运行时数据区域 Java虚拟机在执行java程序时,把内存划分为几个不同的阶段,存在不同的存在时间.不同的用途 先上图 程序计数器:是jvm ...

  10. 从源码角度理清memcache缓存服务

    memcache作为缓存服务器,用来提高性能,大部分互联网公司都在使用.   前言    文章的阅读的对象是中高级开发人员.系统架构师. 本篇文章,不是侧重对memcache的基础知识的总结,比如se ...