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

相关文章

Spring Boot集成/输出/日志级别控制/持久化开发实践

《SpringBoot集成/输出/日志级别控制/持久化开发实践》SpringBoot默认集成Logback,支持灵活日志级别配置(INFO/DEBUG等),输出包含时间戳、级别、类名等信息,并可通过... 目录一、日志概述1.1、Spring Boot日志简介1.2、日志框架与默认配置1.3、日志的核心作用

破茧 JDBC:MyBatis 在 Spring Boot 中的轻量实践指南

《破茧JDBC:MyBatis在SpringBoot中的轻量实践指南》MyBatis是持久层框架,简化JDBC开发,通过接口+XML/注解实现数据访问,动态代理生成实现类,支持增删改查及参数... 目录一、什么是 MyBATis二、 MyBatis 入门2.1、创建项目2.2、配置数据库连接字符串2.3、入

Springboot项目启动失败提示找不到dao类的解决

《Springboot项目启动失败提示找不到dao类的解决》SpringBoot启动失败,因ProductServiceImpl未正确注入ProductDao,原因:Dao未注册为Bean,解决:在启... 目录错误描述原因解决方法总结***************************APPLICA编

深度解析Spring Security 中的 SecurityFilterChain核心功能

《深度解析SpringSecurity中的SecurityFilterChain核心功能》SecurityFilterChain通过组件化配置、类型安全路径匹配、多链协同三大特性,重构了Spri... 目录Spring Security 中的SecurityFilterChain深度解析一、Security

SpringBoot多环境配置数据读取方式

《SpringBoot多环境配置数据读取方式》SpringBoot通过环境隔离机制,支持properties/yaml/yml多格式配置,结合@Value、Environment和@Configura... 目录一、多环境配置的核心思路二、3种配置文件格式详解2.1 properties格式(传统格式)1.

Apache Ignite 与 Spring Boot 集成详细指南

《ApacheIgnite与SpringBoot集成详细指南》ApacheIgnite官方指南详解如何通过SpringBootStarter扩展实现自动配置,支持厚/轻客户端模式,简化Ign... 目录 一、背景:为什么需要这个集成? 二、两种集成方式(对应两种客户端模型) 三、方式一:自动配置 Thick

Spring WebClient从入门到精通

《SpringWebClient从入门到精通》本文详解SpringWebClient非阻塞响应式特性及优势,涵盖核心API、实战应用与性能优化,对比RestTemplate,为微服务通信提供高效解决... 目录一、WebClient 概述1.1 为什么选择 WebClient?1.2 WebClient 与

Java.lang.InterruptedException被中止异常的原因及解决方案

《Java.lang.InterruptedException被中止异常的原因及解决方案》Java.lang.InterruptedException是线程被中断时抛出的异常,用于协作停止执行,常见于... 目录报错问题报错原因解决方法Java.lang.InterruptedException 是 Jav

深入浅出SpringBoot WebSocket构建实时应用全面指南

《深入浅出SpringBootWebSocket构建实时应用全面指南》WebSocket是一种在单个TCP连接上进行全双工通信的协议,这篇文章主要为大家详细介绍了SpringBoot如何集成WebS... 目录前言为什么需要 WebSocketWebSocket 是什么Spring Boot 如何简化 We

java中pdf模版填充表单踩坑实战记录(itextPdf、openPdf、pdfbox)

《java中pdf模版填充表单踩坑实战记录(itextPdf、openPdf、pdfbox)》:本文主要介绍java中pdf模版填充表单踩坑的相关资料,OpenPDF、iText、PDFBox是三... 目录准备Pdf模版方法1:itextpdf7填充表单(1)加入依赖(2)代码(3)遇到的问题方法2:pd