Spring(四):AOP

2024-09-05 08:48
文章标签 java spring aop

本文主要是介绍Spring(四):AOP,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、AOP的概念理解    

        OOP(面向对象编程)使用的是从上到下、纵向的体系结构来解决重复代码的问题,重点关注的是与实际业务联合紧密的模块。而AOP(面向切面编程)则使用的是横向的体系来解决重复代码的问题,它重点关注的是与业务无关,却为业务模块所共同调用(叫做切面Aspect)的逻辑,如执行业务模块的某一功能时,需要记录操作日志、要实现事务,保证业务操作的原子性等等。AOP将这些切面提取出来然后动态的添加到业务逻辑代码之中,即使这些Aspect的实现机制以及代码进行了修改,只需改动一处而不会影响原有业务逻辑代码,从而降低了切面与业务逻辑的耦合度 。 因此,AOP是用来在使用OOP解决问题的过程中增强解决问题的能力,是对OOP的一种补充,实现更好的模块化

二、代理机制

        AOP采用的是代理模式实现的

(1)静态代理

Count.java 

/** * 定义一个账户接口 */  
public interface Count {  // 查看账户方法  public void queryCount();  // 修改账户方法  public void updateCount();  
}  
CountImpl.java 

/** * 委托类(包含业务逻辑) */  
public class CountImpl implements Count {  @Override  public void queryCount() {  System.out.println("查看账户方法...");  }  @Override  public void updateCount() {  System.out.println("修改账户方法...");  }  
}  
CountProxy.java

/** * 这是一个代理类(增强CountImpl实现类) */  
public class CountProxy implements Count {  private CountImpl countImpl;  /** * 覆盖默认构造器 *  * @param countImpl */  public CountProxy(CountImpl countImpl) {  this.countImpl = countImpl;  }  @Override  public void queryCount() {  System.out.println("事务处理之前");  // 调用委托类的方法;  countImpl.queryCount();  System.out.println("事务处理之后");  }  @Override  public void updateCount() {  System.out.println("事务处理之前");  // 调用委托类的方法;  countImpl.updateCount();  System.out.println("事务处理之后");  }  }  
TestCount.java

/** *测试Count类 */  
public class TestCount {  public static void main(String[] args) {  CountImpl countImpl = new CountImpl();  

观察代码可以发现每一个代理类只能为一个接口服务,这样一来程序开发中必然会产生过多的代理,而且,所有的代理操作除了调用的方法不一样之外,其他的操作都一样,则此时肯定是重复代码。解决这一问题最好的做法是可以通过一个代理类完成全部的代理功能,那么此时就必须使用动态代理完成。 

(2)动态代理之JDK

BookFacade.java 

public interface BookFacade {  public void addBook();  
}  
BookFacadeImpl.java 

public class BookFacadeImpl implements BookFacade {  @Override  public void addBook() {  System.out.println("增加图书方法。。。");  }  
}  
BookFacadeProxy.java

import java.lang.reflect.InvocationHandler;  
import java.lang.reflect.Method;  
import java.lang.reflect.Proxy;  
/** * JDK动态代理代理类  */  
public class BookFacadeProxy implements InvocationHandler {  private Object target;  /** * 绑定委托对象并返回一个代理类 * @param target * @return */  public Object bind(Object target) {  this.target = target;  //取得代理对象  return Proxy.newProxyInstance(target.getClass().getClassLoader(),  target.getClass().getInterfaces(), this);   //要绑定接口(这是一个缺陷,cglib弥补了这一缺陷)  }  @Override  /** * 调用目标对象的任何一个方法 都相当于调用invoke(); */  public Object invoke(Object proxy, Method method, Object[] args)  throws Throwable {  Object result=null;  System.out.println("事物开始");  //执行方法  result=method.invoke(target, args);  System.out.println("事物结束");  return result;  }  }  
TestProxy.java 

import net.battier.dao.BookFacade;  
import net.battier.dao.impl.BookFacadeImpl;  
import net.battier.proxy.BookFacadeProxy;  
public class TestProxy {  public static void main(String[] args) {  BookFacadeProxy proxy = new BookFacadeProxy();  BookFacade bookProxy = (BookFacade) proxy.bind(new BookFacadeImpl());  bookProxy.addBook();  }  
}  
JDK的动态代理依靠接口实现,如果有些类并没有实现接口,则不能使用JDK代理,这就要使用cglib动态代理了。

(3)动态代理之CGlib

cglib是针对类来实现代理的,他的原理是对指定的目标类生成一个子类,并覆盖其中方法实现增强。

BookCadeImpl1.java 

/** * 这个是没有实现接口的实现类 */  
public class BookFacadeImpl1 {  public void addBook() {  System.out.println("增加图书的普通方法...");  }  
}  
BookFacadeProxy.java 

import java.lang.reflect.Method;  import net.sf.cglib.proxy.Enhancer;  
import net.sf.cglib.proxy.MethodInterceptor;  
import net.sf.cglib.proxy.MethodProxy;  /** * 使用cglib动态代理 */  
public class BookFacadeCglib implements MethodInterceptor {  private Object target;  /** * 创建代理对象 */  public Object getInstance(Object target) {  this.target = target;  Enhancer enhancer = new Enhancer();  enhancer.setSuperclass(this.target.getClass());  // 回调方法  enhancer.setCallback(this);  // 创建代理对象  return enhancer.create();  }  @Override  // 回调方法  public Object intercept(Object obj, Method method, Object[] args,  MethodProxy proxy) throws Throwable {  System.out.println("事物开始");  proxy.invokeSuper(obj, args);  System.out.println("事物结束");  return null;  }  
}  
Testcglib.java

import net.battier.dao.impl.BookFacadeImpl1;  
import net.battier.proxy.BookFacadeCglib;  
public class TestCglib {  public static void main(String[] args) {  BookFacadeCglib cglib=new BookFacadeCglib();  BookFacadeImpl1 bookCglib=(BookFacadeImpl1)cglib.getInstance(new BookFacadeImpl1());  bookCglib.addBook();  }  
}  
Spring框架中的AOP,如果类实现了接口,就使用JDK的动态代理生成代理对象,如果这个类没有实现任何接口,使用CGLIB生成代理对象.

三、AOP概念


四、Spring中基于AspectJ的AOP

AspectJ是一个基于Java语言的AOP框架,Spring2.0以后新增了对AspectJ切点表达式支持。

AspectJ表达式:
* 语法:execution(表达式)
execution(<访问修饰符>?<返回类型><方法名>(<参数>)<异常>)
       execution(“* cn.itcast.spring3.demo1.dao.*(..)”) ---只检索当前包
       execution(“* cn.itcast.spring3.demo1.dao..*(..)”) ---检索包及当前包的子包.
        execution(* cn.itcast.dao.GenericDAO+.*(..)) ---检索GenericDAO及子类

AspectJ增强:
     @Before 前置通知
     @AfterReturning 后置通知
     @Around 环绕通知
     @AfterThrowing抛出通知
     @After 最终final通知,不管是否异常,该通知都会执行

1、基于注解的AOP

第一步:引入相应jar包.
 spring-aspects-3.2.0.RELEASE.jar
 com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar

第二步:编写被增强的类:
UserDao.java

package cn.itcast.spring3.demo1;

public class UserDao {public void add(){System.out.println("添加用户");}public int update(){System.out.println("修改用户");}public void delete(){System.out.println("删除用户");}public void find(){System.out.println("查询用户");}
}
第三步:使用AspectJ注解形式编写切面类:

/*** 切面类:就是切点与增强结合*/
@Aspect
public class MyAspect {@Before("execution(* cn.itcast.spring3.demo1.UserDao.add(..))")public void before(){System.out.println("前置增强....");}
}
第四步:创建applicationContext.xml

<?xml version="1.0" encoding="UTF-8"?>
<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"><!-- 自动生成代理--><aop:aspectj-autoproxy /><bean id="userDao" class="cn.itcast.spring3.demo1.UserDao"></bean><bean id="myAspect" class="cn.itcast.spring3.demo1.MyAspect"></bean>
</beans>
第五步:测试

package cn.itcast.spring3.demo1;import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
public class SpringTest1 {@Autowired@Qualifier("userDao")private UserDao userDao;@Testpublic void demo1(){userDao.add();}
}

2、基于XML的AOP

第一步:编写被增强的类:

package cn.itcast.spring3.demo2;
public class ProductDao {public int add(){System.out.println("添加商品...");int d = 10/0;return 100;}public void update(){System.out.println("修改商品...");}public void delete(){System.out.println("删除商品...");}public void find(){System.out.println("查询商品...");}
}
第二步:定义切面

package cn.itcast.spring3.demo2;import org.aspectj.lang.ProceedingJoinPoint;/*** 切面类*/
public class MyAspectXML {	public void before(){System.out.println("前置通知...");}public void afterReturing(Object returnVal){System.out.println("后置通知...返回值:"+returnVal);}	public Object around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable{System.out.println("环绕前增强....");Object result = proceedingJoinPoint.proceed();System.out.println("环绕后增强....");return result;}	public void afterThrowing(Throwable e){System.out.println("异常通知..."+e.getMessage());}	public void after(){System.out.println("最终通知....");}
}
第三步:配置applicationContext.xml

<?xml version="1.0" encoding="UTF-8"?>
<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"><!-- 定义被增强的类 --><bean id="productDao" class="cn.itcast.spring3.demo2.ProductDao"></bean><!-- 定义切面 --><bean id="myAspectXML" class="cn.itcast.spring3.demo2.MyAspectXML"></bean><!-- 定义aop配置 --><aop:config><!-- 定义切点: --><aop:pointcut expression="execution(* cn.itcast.spring3.demo2.ProductDao.add(..))" id="mypointcut"/><aop:aspect ref="myAspectXML"><!-- 前置通知 --><aop:before method="before" pointcut-ref="mypointcut"/> <!-- 后置通知 --><aop:after-returning method="afterReturing" pointcut-ref="mypointcut" returning="returnVal"/> <!-- 环绕通知 --><aop:around method="around" pointcut-ref="mypointcut"/> <!-- 异常通知 --><aop:after-throwing method="afterThrowing" pointcut-ref="mypointcut" throwing="e"/> <!-- 最终通知 --><aop:after method="after" pointcut-ref="mypointcut"/></aop:aspect></aop:config>
</beans>

第四步:测试类

package cn.itcast.spring3.demo2;import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext2.xml")
public class SpringTest2 {@Autowired@Qualifier("productDao")private ProductDao productDao;@Testpublic void demo1(){productDao.add();productDao.find();productDao.update();productDao.delete();}
}




这篇关于Spring(四):AOP的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1138489

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

一篇文章彻底搞懂macOS如何决定java环境

《一篇文章彻底搞懂macOS如何决定java环境》MacOS作为一个功能强大的操作系统,为开发者提供了丰富的开发工具和框架,下面:本文主要介绍macOS如何决定java环境的相关资料,文中通过代码... 目录方法一:使用 which命令方法二:使用 Java_home工具(Apple 官方推荐)那问题来了,

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

Java中的.close()举例详解

《Java中的.close()举例详解》.close()方法只适用于通过window.open()打开的弹出窗口,对于浏览器的主窗口,如果没有得到用户允许是不能关闭的,:本文主要介绍Java中的.... 目录当你遇到以下三种情况时,一定要记得使用 .close():用法作用举例如何判断代码中的 input

Spring Gateway动态路由实现方案

《SpringGateway动态路由实现方案》本文主要介绍了SpringGateway动态路由实现方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随... 目录前沿何为路由RouteDefinitionRouteLocator工作流程动态路由实现尾巴前沿S