写在前面:

    首先,我们在编程语言学习的时候,了解过C语言是一种面向过程编程,C++,Java是面向对象编程,而AOP则是面向切面编程。


AOP编程概念及步骤:

    AOP:Aspect object programming,面向切面编程,是指在运行时期,执行核心业务代码时,通过动态代理或Cglib代理的方式,植入关注点代码。AOP是OOP的延续。

    关注点代码:就是指程序中可以复用的代码。

    实现AOP编程有两种方式:

    (1)  注解方式

  (2)XML方式

这篇博客中,我们先来总结一下注解方式实现AOP编程。首先,看一下实现AOP编程的步骤:

    (1)在项目中引入AOP相关Jar

    (2)在bean.xml中引入AOP名称空间


    (3)在bean.xml中开启AOP注解方式

    (4)用注解@Aspect指定AOP类为切面类



    (3)用注解@Pointcut,指定拦截哪些类的哪些方法




AOP编程实现原理

    AOP类,是通过JDK代理或Cglib代理,当拦截到指定类的指定方法时,就会为指定类根据需求创建不同的代理对象。



AOP编程简单案例

下面用一个简单的案例,来实现注解方式的AOP编程

切面类代码:

package aopdemo1;

import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;

@Component
@Aspect // 指定当前类为切面类
public class AOP {
	// 指定切入点表达式,拦截当前包的所有类的save方法
	@Pointcut("execution(* aopdemo1.*.save(..))")
	public void pointcut() {
		System.out.println("开始");
	}

	@Before("pointcut()")
	public void begin() {
		System.out.println("开始");
	}

	@After("pointcut()")
	public void end() {
		System.out.println("结束");
	}
}

IUserDao接口代码



package aopdemo1;

public interface IUserDao {
	public void save();
}

UserDao实现类代码


package aopdemo1;

import org.springframework.stereotype.Component;

@Component //加入IOC容器
public class UserDao implements IUserDao{
	@Override
	public void save() {
		System.out.println("-----User:使用AOP------");
	}
}

另一个StudentDao,但不实现任何接口

package aopdemo1;

import org.springframework.stereotype.Component;

@Component //加入IOC容器
public class StudentDao {

	public void save() {
		System.out.println("-----Student:使用AOP------");
	}
}

APP测试类


package aopdemo1;

import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class APP {
	ApplicationContext ac = new ClassPathXmlApplicationContext("aopdemo1/bean.xml");

	// 有接口的,使用动态代理
	@Test
	public void testApp1() {
		IUserDao userDao = (IUserDao) ac.getBean("userDao");
		userDao.save();
		System.out.println(userDao.getClass());
	}

	// 没有接口的,使用Cglib代理
	@Test
	public void testApp2() {
		StudentDao studentDao = (StudentDao) ac.getBean("studentDao");
		studentDao.save();
		System.out.println(studentDao.getClass());
	}
}

bean.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:p="http://www.springframework.org/schema/p"
	xmlns:context="http://www.springframework.org/schema/context"
	xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx"
	xsi:schemaLocation="http://www.springframework.org/schema/beans
    	 http://www.springframework.org/schema/beans/spring-beans.xsd
     	 http://www.springframework.org/schema/context
         http://www.springframework.org/schema/context/spring-context.xsd
         http://www.springframework.org/schema/aop
         http://www.springframework.org/schema/aop/spring-aop.xsd
         http://www.springframework.org/schema/tx
     	 http://www.springframework.org/schema/tx/spring-tx.xsd">


	<!-- 开启注解扫描 -->
	<context:component-scan base-package="aopdemo1"></context:component-scan>
	<!-- 开启Aop注解方式 -->
	<aop:aspectj-autoproxy proxy-target-class="false"></aop:aspectj-autoproxy>
		
</beans>
程序执行流程说明:

    通过IOC创建一个UserDao或者StudentDao的代理对象,并在AOP类中增加了代理对象的功能,并且通过切入点表达式,指定了拦截哪个类的哪个方法,所以在测试类,当代理对象调用save方法时,不仅仅会输出正常的save方法,还会在save方法的前后,输出AOP的一些功能实现。 

在APP测试类中有两个测试方法,testApp1的结果为

开始
-----User:使用AOP------
结束
class com.sun.proxy.$Proxy10

testApp2的结果为

开始
-----Student:使用AOP------
结束
class aopdemo1.StudentDao$$EnhancerByCGLIB$$1770b917

(1)实现接口的Dao类,会使用动态代理

(2)没有实现接口的Dao类,会使用Cglib代理,生成一个StudentDao的子类作为代理对象。

在AOP类中的几种注解

(1)@Aspect :指定为切面类

(2)@Before :前置通知,在执行目标方法之前执行

(3)@After :后置通知,在执行目标方法之后执行

(4)@AfterReturning :在调用目标方法结束后执行


(5)@AfterThrowing :异常通知,当目标方法异常时,执行关注点代码

注意: 当目标方法出现异常的时候,@AfterReturning下的方法,则不会再执行,大家可以自己测试一下。



(6)@Around :环绕通知,在目标方法的前后都执行



(7)@Pointcut :指定为切入点表达式