查看“Spring:AOP”的源代码
←
Spring:AOP
跳到导航
跳到搜索
因为以下原因,您没有权限编辑本页:
您请求的操作仅限属于该用户组的用户执行:
用户
您可以查看和复制此页面的源代码。
[[category:SpringFramework]] == 关于AOP == Aspect Oriented Programming:面向切面编程,通过预编译方式和运行期间动态代理实现程序功能的统一维护的。 * AOP是OOP的延续。 * 可以在不修改源码的情况下,对程序进行增强; * 应用如:权限校验,日志记录,性能监控,事务控制; == AOP 实现 == Spring 的AOP 的底层用到两种代理机制: * '''JDK 动态代理''':(实现了接口)使用动态代理创建'''接口实现类代理对象'''; * '''CGLIB 动态代理''':(没有实现接口)使用动态代理创建'''类的子类代理对象''';(在子类中调用父类的方法,来完成增强) ** 应用的是底层的字节码增强的技术生成当前类的子类对象; === JDK 动态代理 === * 参见:'''[http://wiki.eijux.com/%E6%A0%B8%E5%BF%83%E6%8A%80%E6%9C%AF%EF%BC%9A%E6%8E%A5%E5%8F%A3%E3%80%81lambda%E8%A1%A8%E8%BE%BE%E5%BC%8F%E4%B8%8E%E5%86%85%E9%83%A8%E7%B1%BB#.E4.BB.A3.E7.90.86 JavaCore:接口(代理)]'''中的代理部分; <syntaxhighlight lang="java"> public class MyJDKProxy implements InvocationHandler { private UserDao userDao; public MyJDKProxy(UserDao userDao) { this.userDao = userDao; } // 编写工具方法:生成代理: public UserDao createProxy() { UserDao userDaoProxy = (UserDao)Proxy.newProxyInstance(userDao.getClass().getClassLoader(), userDao.getClass().getInterfaces(), this); return userDaoProxy; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { if("save".equals(method.getName())) { System.out.println("权限校验================"); } return method.invoke(userDao, args); } } </syntaxhighlight> === CGLIB 动态代理 === * 参见:[http://wiki.eijux.com/%E5%8A%A8%E6%80%81%E4%BB%A3%E7%90%86%E3%80%81CGLIB_%E4%B8%8E_%E5%88%87%E9%9D%A2%E7%BC%96%E7%A8%8B#CGLIB “动态代理、CGLIB 与 切面编程”(CGLIB)] <syntaxhighlight lang="java"> public class MyCglibProxy implements MethodInterceptor { private CustomerDao customerDao; public MyCglibProxy(CustomerDao customerDao) { this.customerDao = customerDao; } // 生成代理的方法: public CustomerDao createProxy() { // 创建Cglib 的核心类: Enhancer enhancer = new Enhancer(); // 设置父类: enhancer.setSuperclass(CustomerDao.class); // 设置回调: enhancer.setCallback(this); // 生成代理: CustomerDao customerDaoProxy = (CustomerDao) enhancer.create(); return customerDaoProxy; } @Override public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable { if("delete".equals(method.getName())) { Object obj = methodProxy.invokeSuper(proxy, args); System.out.println("日志记录================"); return obj; } return methodProxy.invokeSuper(proxy, args); } } </syntaxhighlight> == 关于AspectJ == AspectJ,(Eclipse AspectJ)是一个易用的功能强大的'''AOP框架''';可以单独使用(需要专门的编译器“ajc”),也可以整合到其它框架中。 * 官网:“http://www.eclipse.org/aspectj/”; * AspectJ是AOP编程的完全解决方案,可以做Spring AOP干不了的事情; * AspectJ不是spring一部分,和spring一起使用进行aop操作(Spring2.0以后新增了对AspectJ支持); === AspectJ 与 Spring AOP === AOP编程,存在三种织入方式: # 编译期织入(Compile-time weaving):(编译期、编译后)在Java编译期,采用特殊的'''编译器''',将切面织入到Java类中; #* 编译后:对已经生成的“.class”文件、“jar”包进行织入; # 类加载期织入(Load-time weaving):通过特殊的'''类加载器''',在类字节码加载到JVM时,织入切面; #* 或在JVM启动的时候指定 AspectJ 提供的“agent:-javaagent:xxx/xxx/aspectjweaver.jar”;【?】 # 运行期织入:采用'''CGLib工具'''或'''JDK动态代理'''进行切面的织入; Spring AOP 和 AspectJ 之间的关键区别: {| class="wikitable" ! style="width: 50%"| AspectJ ! style="width: 50%"|Spring AOP |- | 静态织入 * 支持编译时、编译后和加载时织入 | 运行期织入 |- | 支持所有切入点 * (即:可以编织字段、方法、构造函数、静态初始值设定项、最终类/方法等编织) | 仅支持方法执行切入点 * (即:仅支持方法级编织) |- | 使用 Java 编程语言的扩展实现 * 除非设置LTW(Load Time Weaving,加载时织入),否则需要 AspectJ 编译器 (ajc) | 在纯 Java 中实现 *(不需要单独的编译过程) |- | 更好的性能 | 较 AspectJ 慢 |} * Spring AOP使用了Aspect的Annotation,但是并没有使用它的编译器和织入器; *:“@Before”、“@After”等是Aspect的Annotation; * AspectJ在实际运行之前就完成了织入,所以其生成的类没有额外运行时开销。 === 相关术语 === * Joinpoint(连接点):所谓连接点是指那些被拦截到的点。【可以被切入的方法】 ** 在spring 中,这些点指的是'''方法''',因为spring 只支持方法类型的连接点; * '''Pointcut'''(切入点):所谓切入点是指我们要对哪些Joinpoint 进行拦截的定义;【用来切入的方法】 * '''Advice'''(通知/增强):所谓通知是指拦截到Joinpoint 之后所要做的事情就是通知;【切面处理】 *# “前置通知”:; *# “后置通知”:; *# “异常通知”:; *# “最终通知”:无论程序是否正常执行,最终通知的代码会得到执行; *# “环绕通知”:; * Introduction(引介):引介是一种特殊的通知在不修改类代码的前提下, Introduction可以在运行期为类动态地添加一些方法或Field; * '''Target'''(目标对象):代理的目标对象; * '''Weaving'''(织入):是指把增强应用到目标对象来创建新的代理对象的过程;【应用增强生成代理对象】 ** spring采用“动态代理织入”,而AspectJ采用“编译期织入”和“类装载期织入”;【?】 * '''Proxy'''(代理):一个类被AOP 织入增强后,就产生一个结果代理类; * '''Aspect'''(切面): 是切入点和通知(引介)的结合; === jar包 === # spring 的传统AOP 的开发的包: #* “spring-aop-4.2.4.RELEASE.jar” #* “com.springsource.org.aopalliance-1.0.0.jar” * aspectJ 的开发包: #* “com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar” #* “spring-aspects-4.2.4.RELEASE.jar” === 配置文件 === 需要在“applicationContext.xml”中引入AOP相关约束: <syntaxhighlight lang="properties"> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd"> </beans> </syntaxhighlight> == 基于AspectJ:XML配置 == == 基于AspectJ的AOP:注解 ==
返回至“
Spring:AOP
”。
导航菜单
个人工具
登录
命名空间
页面
讨论
大陆简体
已展开
已折叠
查看
阅读
查看源代码
查看历史
更多
已展开
已折叠
搜索
导航
首页
最近更改
随机页面
MediaWiki帮助
笔记
服务器
数据库
后端
前端
工具
《To do list》
日常
阅读
电影
摄影
其他
Software
Windows
WIKIOE
所有分类
所有页面
侧边栏
站点日志
工具
链入页面
相关更改
特殊页面
页面信息