什么是 AOP
AOP:Aspect Oriented Programming
(1)面向切面编程(方面),利用 AOP 可以对业务逻辑的各个部分进行隔离,从而使得
业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
(2)通俗描述:不通过修改源代码方式,在主干功能里面添加新功能
(3)使用登录例子说明 AOP
AOP的底层原理
- 动态代理
(1)有两种情况的动态代理
第一种:有接口的情况,使用 JDK 动态代理
⚫ 创建接口实现类代理对象,增强类的方法
第二种:没有接口情况,使用 CGLIB 动态代理
⚫ 创建子类的代理对象,增强类的方法
JDK 动态代理实现
1、使用 Proxy 类里面的方法创建代理对象
(1)调用 newProxyInstance 方法
方法有三个参数:
第一参数,类加载器
第二参数,增强方法所在的类,这个类实现的接口,支持多个接口
第三参数,实现接口 InvocationHandler,创建代理对象,写增强的部分
2、编写 JDK 动态代理代码
项目结构截图:
(1)创建接口UserDao,定义方法
package com.rqs.spring5.dao;
public interface UserDao {
public int add(int a,int b);
public String update(String id);
}
(2)创建接口实现类UserDaoImpl,实现方法
package com.rqs.spring5.dao;
import org.springframework.stereotype.Repository;
@Repository(value = "userDaoImpl1")
public class UserDaoImpl implements UserDao {
@Override
public int add(int a, int b) {
System.out.println("add未被增强的代码被执行了");
return a+b;
}
@Override
public String update(String id) {
return id;
}
}
(3)使用 Proxy 类创建接口实现类代理对象
package com.rqs.spring5.jdkproxy;
import com.rqs.spring5.dao.UserDao;
import com.rqs.spring5.dao.UserDaoImpl;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.Arrays;
public class JDKProxy {
public static void main(String[] args) {
//创建接口实现类代理对象,不是new出来的,而是通过newProxyInstance方法得来的
Class[] interfaces = {UserDao.class};
UserDaoImpl userDao = new UserDaoImpl();
UserDao dao = (UserDao)Proxy.newProxyInstance(JDKProxy.class.getClassLoader(), interfaces, new UserDaoProxy(userDao));
int result = dao.add(1, 2);
System.out.println("方法执行的结果是:"+result);
}
}
//创建代理对象
class UserDaoProxy implements InvocationHandler {
// 创建的是谁的代理对象,就把谁传递过来,可以通过有参构造的方式传递
// public UserDaoProxy(UserDaoImpl userDao) {
private Object object;
public UserDaoProxy(Object object) {//参数写成Object 更通用
this.object = object;
}
// 写需要增强的部分,本例是在add方法的【return a+b;】之前和之后分别做一些其它操作
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//方法之前
System.out.println(method.getName()+"方法之前执行的代码..."+"传递的参数是:"+ Arrays.toString(args));
//被增强的方法,本例是add方法
Object res = method.invoke(object, args);
//方法之后
System.out.println(method.getName()+"方法之后执行的代码...输出创建的代理对象:"+object);
return res;
}
}
执行main方法的结果: