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将各种数据写入Excel表格的操作示例

《使用Java将各种数据写入Excel表格的操作示例》在数据处理与管理领域,Excel凭借其强大的功能和广泛的应用,成为了数据存储与展示的重要工具,在Java开发过程中,常常需要将不同类型的数据,本文... 目录前言安装免费Java库1. 写入文本、或数值到 Excel单元格2. 写入数组到 Excel表格

Java并发编程之如何优雅关闭钩子Shutdown Hook

《Java并发编程之如何优雅关闭钩子ShutdownHook》这篇文章主要为大家详细介绍了Java如何实现优雅关闭钩子ShutdownHook,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起... 目录关闭钩子简介关闭钩子应用场景数据库连接实战演示使用关闭钩子的注意事项开源框架中的关闭钩子机制1.

Maven中引入 springboot 相关依赖的方式(最新推荐)

《Maven中引入springboot相关依赖的方式(最新推荐)》:本文主要介绍Maven中引入springboot相关依赖的方式(最新推荐),本文给大家介绍的非常详细,对大家的学习或工作具有... 目录Maven中引入 springboot 相关依赖的方式1. 不使用版本管理(不推荐)2、使用版本管理(推

Java 中的 @SneakyThrows 注解使用方法(简化异常处理的利与弊)

《Java中的@SneakyThrows注解使用方法(简化异常处理的利与弊)》为了简化异常处理,Lombok提供了一个强大的注解@SneakyThrows,本文将详细介绍@SneakyThro... 目录1. @SneakyThrows 简介 1.1 什么是 Lombok?2. @SneakyThrows

在 Spring Boot 中实现异常处理最佳实践

《在SpringBoot中实现异常处理最佳实践》本文介绍如何在SpringBoot中实现异常处理,涵盖核心概念、实现方法、与先前查询的集成、性能分析、常见问题和最佳实践,感兴趣的朋友一起看看吧... 目录一、Spring Boot 异常处理的背景与核心概念1.1 为什么需要异常处理?1.2 Spring B

如何在 Spring Boot 中实现 FreeMarker 模板

《如何在SpringBoot中实现FreeMarker模板》FreeMarker是一种功能强大、轻量级的模板引擎,用于在Java应用中生成动态文本输出(如HTML、XML、邮件内容等),本文... 目录什么是 FreeMarker 模板?在 Spring Boot 中实现 FreeMarker 模板1. 环

SpringMVC 通过ajax 前后端数据交互的实现方法

《SpringMVC通过ajax前后端数据交互的实现方法》:本文主要介绍SpringMVC通过ajax前后端数据交互的实现方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价... 在前端的开发过程中,经常在html页面通过AJAX进行前后端数据的交互,SpringMVC的controll

Java中的工具类命名方法

《Java中的工具类命名方法》:本文主要介绍Java中的工具类究竟如何命名,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录Java中的工具类究竟如何命名?先来几个例子几种命名方式的比较到底如何命名 ?总结Java中的工具类究竟如何命名?先来几个例子JD

Java Stream流使用案例深入详解

《JavaStream流使用案例深入详解》:本文主要介绍JavaStream流使用案例详解,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录前言1. Lambda1.1 语法1.2 没参数只有一条语句或者多条语句1.3 一个参数只有一条语句或者多

Spring Security自定义身份认证的实现方法

《SpringSecurity自定义身份认证的实现方法》:本文主要介绍SpringSecurity自定义身份认证的实现方法,下面对SpringSecurity的这三种自定义身份认证进行详细讲解,... 目录1.内存身份认证(1)创建配置类(2)验证内存身份认证2.JDBC身份认证(1)数据准备 (2)配置依