AOP相关操作

AOP

基本概念

面向切面编程,可以对业务逻辑各个部分进行隔离,从而使得业务逻辑各个部分耦合度降低,提高可重用性

从业务逻辑代码中分离开来。

不通过修改原代码的方式添加新的功能权限判断模块

底层原理

AOP使用到了动态代理

1.有两种情况的动态代理

一、有接口的情况,使用JDK中的动态代理

创建接口实现类代理对象,增强类的方法

interface UserDao{
    public void login();
}

class UserImpl implements UserDao{
    public void login();
}

JDK的动态代理
    1、创建UserDao接口实现类代理对象


二、没有接口的情况,使用CGLIB动态代理

创建子类的代理对象,用来增强

class User{
    public void add()
    {
        .....
    }
}
//原始方法
class Person extends User{
    public void add()
    {
        super.add;
        增强逻辑
    }
}

//CGLIB动态代理
//创建当前类子类的代理对象
//代理对象
AOP JDK动态代理
//使用Proxy类中的一个方法创建代理对象。
//newProxyInstance方法
//三个参数
//1、classLoad类加载器
//2、增强方法所在的类,这个类实现的接口,可以是多个接口
//3、实现这个接口InvocationHandler,创建代理对象,写增强部分。

//创建接口
public interface UserDao
{
    public int add(int a, int b);
    
    public String update(String id);
}
//创建接口实现类

public class UserDaoImpl implements UserDao
{
    @Override
    public int add(int a, int b)
    {
        return a + b;
    }
    
    
    @Override
    public String update(String id)
    {
        return id;
    }
}

//增强部分
public class JDKProxy
{
    public static void main(String[] args)
    {
       	Class[] interface = {UserDao.class}
        //Proxy.newProxyInstance(JDKProxy.class.getClassLoad, interfaces, )
        UserDaoImpl userDao = new UserDaoImpl();
        UserDao dao = (UserDao)Proxy.newProxyInstance(JDKProxy.class.getClassLoad, interfaces, new UserDaoProxy(userDao));
        int add = dao.add(1, 2);
        System.out.pritnln(add);
    }
}

class UserDaoProxy implements InvocationHandler
{
    //把创建的是谁的代理对象,把谁传递过来
    //有参构造进行传递
    private Objecct obj;
    public UserDaoProxy(Object obj)
    {
        this.obj = obj;
    }
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable
    {
        //方法之前
        System.out.println("方法之前。。。。"  + method.getName());
        
        //被增强的方法执行
        Object res = method.invoke(obj, args);
        
        //方法之后
        System.out.println("方法之后执行" + obj);
        return null;
    }
}
AOP术语

1、连接点

哪些方法被增强,这些方方法就是连接点

2、切入点
实际被真正被增强的方法,成为切入点

3、通知(增强)

实际增强的逻辑部分称为通知(增强)

通知有多种类型

1、前置通知

2、后置通知

3、环绕通知

4、异常通知

5、最终通知 finally

4、切面

是动作

1把通知应用到切入点的过程

class User
{
    add();
    update();
    select();
    delete();
}

AOP操作(准备)

1、Spring框架一般基于AspectJ实现AOP操作

AspectJ

不是Spring组成部分,独立于AOP框架,一般一起使用,进行aop操作

2、基于AspectJ实现AOP操作

  • XML文件实现

  • 注解方式实现(一般情况下)

引入依赖

4、切入点表达式

知道哪个类里面的哪个方法进行增强

语法结构

execution([权限修饰符] [返回类型] [类全路径] [方法名称] ([参数列表]))

execution(* com.z.dao.BookDao.add(…))

execution(* com.z.dao.BookDao.*(…))

AOP操作(AspectJ注解)

//创建类
@Component
public class User{
    public void add()
    {
        System.out.println("add.....");
    }
}

//创建增强类
//在增强类里面增加方法,代表不同的通知类型


//前置通知
@Component
@Aspect
public class UserProxy
{
    @Before(value="excution(* com.*(..))")
    public void before()
    {
        System.out.println("before..... ");
    }
    
    @After(value="excution(* com.*(..))")
    
    @AfterReturning(value="excution(* com.*(..))")
    
    
  	//异常通知
    @AfterThrowing(value="excution(* com.*(..))")
    
    @Around(value="excution(* com.*(..))")
    public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable
    {
        System.out.println("环绕之前...");
        proceedingJoinPoint.proceed();
        System.out.println("环绕之后")
    }
}
//在sring配置文件中,开启注解扫描
//使用注解创建User和UserProxy对象
//在增强类上面添加注解@Aspect
//开启扫描
//配置不同通知
    //在增强类的里面
开启生成代理对象
<aop:aspectj-autoproxy></aop:aspectj-autoproxy>
@Test
public void testAopAnno()
{
	ApplicationContext context = new ClassPathXlApplication("bean1.xml");
    User user = context.getBean("user", User.class);
    user.add();
}

5、细节问题

公共切入点进行抽取

@Pointcut(value="execution(* com.*(..))")
public void pointdemo()
{
    
}

@Before(value = "pointdemo()")
public void before()
{
    ....
}

6、有多个增强类对同一个方法进行增强,可以设置优先级

在增强类上添加注解@Order(数字)数字值越小,优先级越高

@AspectJ
@Component
@Order(0)
public class a{
	@Before(value = "execution()")
    public void before()
    {
        System.out.println("....");
    }
}
### AOP实现操作记录的方法 Spring AOP使用代理模式和字节码操作技术实现AOP功能。为目标对象创建代理对象,当调用目标对象的方法时,实际上调用的是代理对象的方法。代理对象在调用目标方法前后,插入额外的代码(即切面的通知)来实现AOP功能。具体使用字节码操作技术为目标对象生成一个子类(使用CGLIB代理)或接口的实现类(使用JDK动态代理),并在该类中重写目标方法,以插入额外的代码[^2]。 AOP有五种通知类型可用于操作记录: - 前置通知:在目标方法调用之前执行,可以获得切入点信息。 - 后置通知:在目标方法执行后执行,目标方法有异常时不执行。 - 异常通知:在目标方法抛出异常时执行,可以获取异常信息。 - 最终通知:在目标方法执行后执行,无论是否有异常都执行。 - 环绕通知:最强大的通知类型,在目标方法执行前后操作,可以阻止目标方法执行[^1]。 ### 示例代码 以下是一个使用Spring AOP进行操作记录的示例: 首先,添加Spring AOP依赖(以Maven为例): ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-aop</artifactId> </dependency> ``` 定义一个切面类来记录操作日志: ```java import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.After; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Pointcut; import org.springframework.stereotype.Component; @Aspect @Component public class OperationLogAspect { // 定义切入点,这里以Service层的所有方法为例 @Pointcut("execution(* com.example.service.*.*(..))") public void serviceMethods() {} // 后置通知,在目标方法执行后记录操作日志 @After("serviceMethods()") public void logOperation(JoinPoint joinPoint) { String methodName = joinPoint.getSignature().getName(); Object[] args = joinPoint.getArgs(); System.out.println("执行了方法:" + methodName + ",参数:" + java.util.Arrays.toString(args)); } } ``` 定义一个Service类: ```java package com.example.service; import org.springframework.stereotype.Service; @Service public class UserService { public void addUser(String username) { System.out.println("添加用户:" + username); } } ``` ### 相关技术 - **参数解析**:可以使用特定的解析模板来获取入参和返回对象的属性值。例如,`#args.对象名.属性名` 或 `#args.属性名` 用于获取入参数里面属性值,`#return.对象名.属性名` 或 `#return.属性名` 用于获取返回对象的属性值。还有高级用法,如 `return['data'].username` 可获取返回值map里面的 `data` 键的值,再拿到对象获取用户名 `username` [^3]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值