1.1 什么是AOP
AOP意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生泛型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同事提高了开发效率。
1.2Spring中AOP三种实现方式
方式一:使用spring的api接口 【主要是spring api 接口实现】
方法二:自定义来实现aop
方法三:使用注解实现aop
spring aop (面向切面)常用于数据库事务中,使用了2种代理。
- jdk动态代理:对实现了接口的类生成代理对象。要使用jdk动态代理,要求类必须要实现接口。
- cglib代理:对类生成代理对象。
1.3事例解析
以往我们开发项目都是纵向开发,dao层——>service——>controller。现在通过添加日志横向开发日志功能,实现不改变之前代码的前提下开发!
1.4 使用spring的api接口来实现AOP
(1)结构图
(2)UserService接口
package com.li.service;
public interface UserService {
public void add();
public void delete();
public void update();
public void select();
}
(3)UserServiceImp类
package com.li.service;
/**
* @Description:
* @Author: lizq
* @date 2021/4/6 11:54
**/
public class UserServiceImp implements UserService{
public void add() {
System.out.println("增加了一个用户!");
}
public void delete() {
System.out.println("删除了一个用户!");
}
public void update() {
System.out.println("修改了一个用户!");
}
public void select() {
System.out.println("查询了一个用户!");
}
}
(3)Log类
package com.li.log;
import org.springframework.aop.MethodBeforeAdvice;
import java.lang.reflect.Method;
/**
* @Description:
* @Author: lizq
* @date 2021/4/6 13:36
**/
public class Log implements MethodBeforeAdvice {
//method :要执行的目标对象的方法
//args : 参数
//target :目标对象
public void before(Method method, Object[] args, Object target) throws Throwable {
System.out.println(target.getClass().getName()+"的"+method.getName()+"被执行了");
}
}
(4)Afterlog类
package com.li.log;
import org.springframework.aop.AfterReturningAdvice;
import java.lang.reflect.Method;
/**
* @Description:
* @Author: lizq
* @date 2021/4/6 13:44
**/
public class Afterlog implements AfterReturningAdvice {
public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {
System.out.println("执行了"+method.getName()+"方法,返回结果为:"+returnValue);
}
}
(5)beans.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
https://www.springframework.org/schema/aop/spring-aop.xsd">
<!-- 使用spring来创建对象,在spring这些都称为bean -->
<bean id="userService" class="com.li.service.UserServiceImp"></bean>
<bean id="log" class="com.li.log.Log"></bean>
<bean id="afterlog" class="com.li.log.Afterlog"></bean>
<!--方法一:使用原生spring API接口-->
<!--配置aop-->
<aop:config>
<!--切入点:expression:表达式,execution(要执行的位置!.* 该方法下的任意参数(..))-->
<aop:pointcut id="pointcut" expression="execution(* com.li.service.UserServiceImp.*(..))"/>
<aop:advisor advice-ref="log" pointcut-ref="pointcut"></aop:advisor>
<aop:advisor advice-ref="afterlog" pointcut-ref="pointcut"></aop:advisor>
</aop:config>
</beans>
(6)MyTest类
import com.li.log.Afterlog;
import com.li.service.UserService;
import com.li.service.UserServiceImp;
import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
* @Description:
* @Author: lizq
* @date 2021/4/6 14:13
**/
public class MyTest {
public static void main(String[] args) {
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
//动态代理的是接口
UserService userService = context.getBean("userService", UserService.class);
userService.add();
}
}
(7)执行MyTest输出结果
com.li.service.UserServiceImp的add被执行了
增加了一个用户!
执行了add方法,返回结果为:null
(8)总结
本章节通过使用spring的api接口来横向开发日志功能,写的不好的地方请各位同仁多多指教!其他两种方式后续更新~