栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > Java

Spring之AOP

Java 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

Spring之AOP

目录

1、AOP概述

2、AOP底层原理

3、AOP术语

 4、AOP 操作

4.1、基于 AspectJ 实现 AOP 操作

4.2、切入点表达式

4.3、AOP 操作(AspectJ 注解)

4.4、AOP 操作(配置文件)


1、AOP概述

Spring 有两个核心部分:IOC 和 Aop

(1)IOC:控制反转,把创建对象过程交给 Spring 进行管理

(2)AOP:面向切面编程,不修改源代码进行功能增强

 AOP:AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。

何为AOP:用一个用户登录的例子来说明:

用户登录一般过程:1.收到用户的用户名和密码。2.数据库取查询该用户名和密码。3. 判断数据库是否存在该用户名和密码。

若要在此基础上再加一个用户权限。不同的人可以有不同的管理权限。我们可以使用注解或者其它的方式在判断登录成功的模块后,再添加一个新的模块。

 

 这样做的好处,不需要修改之前我们写过的代码,并且若在以后不需要判断权限,我们只需要将判断程序权限的模块移除。也就是AOP通俗描述:不通过修改源代码方式,在主干功能里面添加新功能

2、AOP底层原理

AOP的底层使用的是动态代理。

1、有接口情况,使用 JDK 动态代理

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

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

 通过接口实现类的代理对象,再通过代理对象把功能加强

代理对象和实现类对象区别:实现类对象是new出来的,而代理对象 不需要new但是能做和实现类相同的功能,和接口实现类是等价的。

2. 没有接口情况,使用 CGLIB 动态代理

创建子类的代理对象,增强类的方法

 CGLIB创建子类的代理对象,和实现类等价,和实现类是等价的。

Spring5里面动态代理的代码Spring已经做好了封装

3、AOP的实现(使用JDK的动态代理)

使用 JDK 动态代理,使用 Proxy 类里面的方法创建代理对象

Proxy 类在Java.lang包下

调用newProxyInstance有三个参数

newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h)

第一参数,类加载器,用哪个类加载器去加载代理对象

第二参数,动态代理类需要实现的接口,支持多个接口

第三参数,实现这个接口 InvocationHandler,创建代理对象,写增强的部分,动态代理方法在执行时,会调用InvocationHandler里面的invoke方法去执行

例子实现:

package dao;

import org.springframework.stereotype.Component;

public interface UserDao {
    public int add(int a,int b);

    public  String update(String Id);

}
package dao;

import org.springframework.stereotype.Component;

@Component
public class UserDao1Impl implements UserDao {

    @Override
    public int add(int a, int b) {
        return a + b;
    }

    @Override
    public String update(String Id) {
        return Id;
    }
}
package dao;

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) {
        Class[] interfaces = {UserDao.class};//Class数组类型
        UserDao1Impl userDao = new UserDao1Impl();
        Proxy.newProxyInstance(JDKProxy.class.getClassLoader(), interfaces, new UserDaoProxy(userDao));//下面UserDaoProxy方法实现了InvocationHandler,并且加了一个有参构造
        int result = userDao.add(1, 2);
        System.out.println("result:"+result);
    }
}

class UserDaoProxy implements InvocationHandler {

    private Object obj;
    public UserDaoProxy(Object obj) {
        this.obj = obj;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        //proxy代理对象 method当前方法 args参数
        System.out.println("方法之前执行...." + method.getName() + " :传递的参数..." + Arrays.toString(args));//Arrays.toString(args)数组转字符串
        //被增强的方法执行
        Object res = method.invoke(obj, args);//当前的方法
        //方法之后
        System.out.println("方法之后执行...." + obj);
        return res;
    }
}

3、AOP术语

连接点(Joinpoint):类中哪些方法可以被增强

切入点(Pointcut):实际上真正被增强了的方法

通知、增强(Advice):你想要的功能,在特定的连接点,AOP框架执行的动作。
Spring切面可以应用5种类型的通知:
前置通知(Before):在目标方法被调用之前调用通知功能;
后置通知(After):在目标方法完成之后调用通知,此时不会关心方法的输出是什么;
返回通知(After-returning):在目标方法成功执行之后调用通知;
异常通知(After-throwing):在目标方法抛出异常后调用通知;
环绕通知(Around):通知包裹了被通知的方法,在被通知的方法调用之前和调用之后执行自定义的行为。

目标(Target):被通知的对象(方法)

代理(Proxy):向目标对象应用通知之后创建的对象

切面:一种动作,把我们的通知应用到切入点的过程

 4、AOP 操作

Spring 框架一般都是基于 AspectJ 实现 AOP 操作

AspectJ 不是 Spring 组成部分,独立 AOP 框架,一般把 AspectJ 和 Spirng 框架一起使 用,进行 AOP 操作

4.1、基于 AspectJ 实现 AOP 操作

(1)基于 xml 配置文件实现

(2)基于注解方式实现

4.2、切入点表达式

(1)切入点表达式作用:知道对哪个类里面的哪个方法进行增强

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

eg 1:对 com.kk.dao.BookDao 类里面的 add 进行增强

 execution(* com.kk.dao.BookDao.add(..))

eg 2:对 com.kk.dao.BookDao 类里面的所有的方法进行增强

execution(* com.kk.dao.BookDao.* (..))

eg 3:对 com.kk.dao 包里面所有类,类里面所有方法进行增强

execution(* com.kk.dao.*.* (..)) 

4.3、AOP 操作(AspectJ 注解)

1、创建类,在类里面定义方法

public class User {
 public void add() {
 System.out.println("add.......");
 }
}

2、创建增强类(编写增强逻辑)

(1)在增强类里面,创建方法,让不同方法代表不同通知类型

        假如我想让这个方法在add方法前输出(UserProxy类为例)

//增强的类
public class UserProxy {
 public void before() {//前置通知
 System.out.println("before......");
 }
}

3、进行通知的配置

(1)在 spring 配置文件中,先开启注解扫描。可以写配置类(完全注解开发),或者配置文件。




    
    

 将配置文件

xmlns="http://www.springframework.org/schema/beans"

中添加context和aop

http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd

将beans分别改成context和aop,context用来创建对象,在bean管理中有学。

开启注解扫描 

(2)使用注解创建 User 和 UserProxy 对象,在类的上面加 @Component 注解

(3)在增强类上面添加注解 @Aspect生成代理对象

 (4)在 spring 配置文件中开启生成代理对象

通俗点来说这个配置文件就是到你的类上面,找@Aspect注解,就生成代理对象

 

4、配置不同类型的通知(有5中类型)

(1)在增强类的里面,在作为通知方法上面添加通知类型注解,使用切入点表达式配置

package AOP;

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
import org.springframework.stereotype.Component;

@Component  //给类创建对象
@Aspect  //生成代理对象

public class UserProxy {
    //前置通知
//@Before 注解表示作为前置通知
    @Before(value = "pointdemo()")
    public void before() {
        System.out.println("before.........");
    }

    //后置通知(返回通知)
    @AfterReturning(value = "execution(* AOP.User.add(..))")
    public void afterReturning() {
        System.out.println("afterReturning.........");
    }
    //最终通知
    @After(value = "execution(* AOP.User.add(..))")
    public void after() {
        System.out.println("after.........");
    }
    //异常通知
    @AfterThrowing(value = "execution(* AOP.User.add(..))")
    public void afterThrowing() {
        System.out.println("afterThrowing.........");
    }
    //环绕通知
    @Around(value = "execution(* AOP.User.add(..))")
    public void around(ProceedingJoinPoint proceedingJoinPoint) throws
            Throwable {
        System.out.println("环绕之前.........");
        //被增强的方法执行
        proceedingJoinPoint.proceed();
        System.out.println("环绕之后.........");
    }
    @Pointcut("execution(* AOP.User.add(..))")
    public void pointdemo() {
    }


}

5、相同的切入点抽取

 //前置通知 //@Before 注解表示作为前置通知

 

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

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

@Component
@Aspect
@Order(1)
public class PersonProxy{}

7、完全使用注解开发

(1)创建配置类,不需要创建 xml 配置文件

package config;


import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;

@Configuration
@ComponentScan(basePackages = {"AOP"})
@EnableAspectJAutoProxy(proxyTargetClass = true)
public class SpringConfig {
}

注意与配置文件不同的是

ApplicationContext contest=new AnnotationConfigApplicationContext(SpringConfig.class); 这里和创建完全注解Bean的是一样的

package AOP;

import config.SpringConfig;
import org.junit.jupiter.api.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class TestDemo {
    @Test

    public void test(){

        ApplicationContext contest=new AnnotationConfigApplicationContext(SpringConfig.class);
        User user = contest.getBean("user", User.class);
        user.add();
    }

}

 

4.4、AOP 操作(配置文件)

实际开发中一般都使用的是注解的方式,配置文件的方式相对于注解方式会更加的麻烦。

方法简要说一下:

1、创建两个类,增强类和被增强类,创建方法

2、在 spring 配置文件中创建两个类对象


3、在 spring 配置文件中配置切入点


 
 
 
 
 
 
 

转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/338140.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号