😧 Spring_day03(五)

简介: 😧 Spring_day03

4.4.2 获取参数


非环绕通知获取方式


在方法上添加JoinPoint,通过JoinPoint来获取参数


@Component
@Aspect
public class MyAdvice {
    @Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")
    private void pt(){}
    @Before("pt()")
    public void before(JoinPoint jp) 
        Object[] args = jp.getArgs();
        System.out.println(Arrays.toString(args));
        System.out.println("before advice ..." );
    }
  //...其他的略
}

运行App类,可以获取如下内容,说明参数100已经被获取


网络异常,图片无法展示
|


思考:方法的参数只有一个,为什么获取的是一个数组?


因为参数的个数是不固定的,所以使用数组更通配些。


如果将参数改成两个会是什么效果呢?


(1)修改BookDao接口和BookDaoImpl实现类

public interface BookDao {
    public String findName(int id,String password);
}
@Repository
public class BookDaoImpl implements BookDao {
    public String findName(int id,String password) {
        System.out.println("id:"+id);
        return "itcast";
    }
}


(2)修改App类,调用方法传入多个参数

public class App {
    public static void main(String[] args) {
        ApplicationContext ctx = new AnnotationConfigApplicationContext(SpringConfig.class);
        BookDao bookDao = ctx.getBean(BookDao.class);
        String name = bookDao.findName(100,"itheima");
        System.out.println(name);
    }
}

(3)运行App,查看结果,说明两个参数都已经被获取到


网络异常,图片无法展示
|


说明:


使用JoinPoint的方式获取参数适用于前置后置返回后抛出异常后通知。剩下的大家自行去验证。


环绕通知获取方式


环绕通知使用的是ProceedingJoinPoint,因为ProceedingJoinPoint是JoinPoint类的子类,所以对于ProceedingJoinPoint类中应该也会有对应的getArgs()方法,我们去验证下:

@Component
@Aspect
public class MyAdvice {
    @Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")
    private void pt(){}
    @Around("pt()")
    public Object around(ProceedingJoinPoint pjp)throws Throwable {
        Object[] args = pjp.getArgs();
        System.out.println(Arrays.toString(args));
        Object ret = pjp.proceed();
        return ret;
    }
  //其他的略
}


运行App后查看运行结果,说明ProceedingJoinPoint也是可以通过getArgs()获取参数


网络异常,图片无法展示
|


注意:


  • pjp.proceed()方法是有两个构造方法,分别是:
    网络异常,图片无法展示
    |
  • 调用无参数的proceed,当原始方法有参数,会在调用的过程中自动传入参数
  • 所以调用这两个方法的任意一个都可以完成功能
  • 但是当需要修改原始方法的参数时,就只能采用带有参数的方法,如下:
@Component
@Aspect
public class MyAdvice {
    @Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")
    private void pt(){}
    @Around("pt()")
    public Object around(ProceedingJoinPoint pjp) throws Throwable{
        Object[] args = pjp.getArgs();
        System.out.println(Arrays.toString(args));
        args[0] = 666;
        Object ret = pjp.proceed(args);
        return ret;
    }
  //其他的略
}

有了这个特性后,我们就可以在环绕通知中对原始方法的参数进行拦截过滤,避免由于参数的问题导致程序无法正确运行,保证代码的健壮性。


4.4.3 获取返回值


对于返回值,只有返回后AfterReturing和环绕Around这两个通知类型可以获取,具体如何获取?


环绕通知获取返回值


@Component
@Aspect
public class MyAdvice {
    @Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")
    private void pt(){}
    @Around("pt()")
    public Object around(ProceedingJoinPoint pjp) throws Throwable{
        Object[] args = pjp.getArgs();
        System.out.println(Arrays.toString(args));
        args[0] = 666;
        Object ret = pjp.proceed(args);
        return ret;
    }
  //其他的略
}

上述代码中,ret就是方法的返回值,我们是可以直接获取,不但可以获取,如果需要还可以进行修改。


返回后通知获取返回值


@Component
@Aspect
public class MyAdvice {
    @Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")
    private void pt(){}
    @AfterReturning(value = "pt()",returning = "ret")
    public void afterReturning(Object ret) {
        System.out.println("afterReturning advice ..."+ret);
    }
  //其他的略
}

注意:


(1)参数名的问题


网络异常,图片无法展示
|


(2)afterReturning方法参数类型的问题


参数类型可以写成String,但是为了能匹配更多的参数类型,建议写成Object类型


(3)afterReturning方法参数的顺序问题


网络异常,图片无法展示
|


运行App后查看运行结果,说明返回值已经被获取到


网络异常,图片无法展示
|


4.4.4 获取异常


对于获取抛出的异常,只有抛出异常后AfterThrowing和环绕Around这两个通知类型可以获取,具体如何获取?


环绕通知获取异常


这块比较简单,以前我们是抛出异常,现在只需要将异常捕获,就可以获取到原始方法的异常信息了

@Component
@Aspect
public class MyAdvice {
    @Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")
    private void pt(){}
    @Around("pt()")
    public Object around(ProceedingJoinPoint pjp){
        Object[] args = pjp.getArgs();
        System.out.println(Arrays.toString(args));
        args[0] = 666;
        Object ret = null;
        try{
            ret = pjp.proceed(args);
        }catch(Throwable throwable){
            t.printStackTrace();
        }
        return ret;
    }
  //其他的略
}

在catch方法中就可以获取到异常,至于获取到异常以后该如何处理,这个就和你的业务需求有关了。


抛出异常后通知获取异常
@Component
@Aspect
public class MyAdvice {
    @Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")
    private void pt(){}
    @AfterThrowing(value = "pt()",throwing = "t")
    public void afterThrowing(Throwable t) {
        System.out.println("afterThrowing advice ..."+t);
    }
  //其他的略
}


如何让原始方法抛出异常,方式有很多,

@Repository
public class BookDaoImpl implements BookDao {
    public String findName(int id,String password) {
        System.out.println("id:"+id);
        if(true){
            throw new NullPointerException();
        }
        return "itcast";
    }
}

注意:


网络异常,图片无法展示
|


运行App后,查看控制台,就能看的异常信息被打印到控制台


网络异常,图片无法展示
|


至此,AOP通知如何获取数据就已经讲解完了,数据中包含参数返回值异常(了解)


4.5 百度网盘密码数据兼容处理


4.5.1 需求分析


需求: 对百度网盘分享链接输入密码时尾部多输入的空格做兼容处理。


网络异常,图片无法展示
|


问题描述:


  • 点击链接,会提示,请输入提取码,如下图所示
    网络异常,图片无法展示
    |
  • 当我们从别人发给我们的内容中复制提取码的时候,有时候会多复制到一些空格,直接粘贴到百度的提取码输入框
  • 但是百度那边记录的提取码是没有空格的
  • 这个时候如果不做处理,直接对比的话,就会引发提取码不一致,导致无法访问百度盘上的内容
  • 所以多输入一个空格可能会导致项目的功能无法正常使用。
  • 此时我们就想能不能将输入的参数先帮用户去掉空格再操作呢?


答案是可以的,我们只需要在业务方法执行之前对所有的输入参数进行格式处理——trim()


  • 是对所有的参数都需要去除空格么?


也没有必要,一般只需要针对字符串处理即可。


  • 以后涉及到需要去除前后空格的业务可能会有很多,这个去空格的代码是每个业务都写么?


可以考虑使用AOP来统一处理。


  • AOP有五种通知类型,该使用哪种呢?


我们的需求是将原始方法的参数处理后在参与原始方法的调用,能做这件事的就只有环绕通知。


综上所述,我们需要考虑两件事:
①:在业务方法执行之前对所有的输入参数进行格式处理——trim()
②:使用处理后的参数调用原始方法——环绕通知中存在对原始方法的调用


4.5.2 环境准备


  • 创建一个Maven项目
  • pom.xml添加Spring依赖
<dependencies>
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-context</artifactId>
      <version>5.2.10.RELEASE</version>
    </dependency>
    <dependency>
      <groupId>org.aspectj</groupId>
      <artifactId>aspectjweaver</artifactId>
      <version>1.9.4</version>
    </dependency>
  </dependencies>


  • 添加ResourcesService,ResourcesServiceImpl,ResourcesDao和ResourcesDaoImpl类


public interface ResourcesDao {
    boolean readResources(String url, String password);
}
@Repository
public class ResourcesDaoImpl implements ResourcesDao {
    public boolean readResources(String url, String password) {
        //模拟校验
        return password.equals("root");
    }
}
public interface ResourcesService {
    public boolean openURL(String url ,String password);
}
@Service
public class ResourcesServiceImpl implements ResourcesService {
    @Autowired
    private ResourcesDao resourcesDao;
    public boolean openURL(String url, String password) {
        return resourcesDao.readResources(url,password);
    }
}


  • 创建Spring的配置类
@Configuration
@ComponentScan("com.itheima")
public class SpringConfig {
}


  • 编写App运行类
public class App {
    public static void main(String[] args) {
        ApplicationContext ctx = new AnnotationConfigApplicationContext(SpringConfig.class);
        ResourcesService resourcesService = ctx.getBean(ResourcesService.class);
        boolean flag = resourcesService.openURL("http://pan.baidu.com/haha", "root");
        System.out.println(flag);
    }
}

最终创建好的项目结构如下:


网络异常,图片无法展示
|


现在项目的效果是,当输入密码为"root"控制台打印为true,如果密码改为"root  "控制台打印的是false


需求是使用AOP将参数进行统一处理,不管输入的密码root前后包含多少个空格,最终控制台打印的都是true。


4.5.3 具体实现


步骤1:开启SpringAOP的注解功能
@Configuration
@ComponentScan("com.itheima")
@EnableAspectJAutoProxy
public class SpringConfig {
}


步骤2:编写通知类
@Component
@Aspect
public class DataAdvice {
    @Pointcut("execution(boolean com.itheima.service.*Service.*(*,*))")
    private void servicePt(){}
}
步骤3:添加环绕通知
@Component
@Aspect
public class DataAdvice {
    @Pointcut("execution(boolean com.itheima.service.*Service.*(*,*))")
    private void servicePt(){}
    @Around("DataAdvice.servicePt()")
    // @Around("servicePt()")这两种写法都对
    public Object trimStr(ProceedingJoinPoint pjp) throws Throwable {
        Object ret = pjp.proceed();
        return ret;
    }
}
步骤4:完成核心业务,处理参数中的空格
@Component
@Aspect
public class DataAdvice {
    @Pointcut("execution(boolean com.itheima.service.*Service.*(*,*))")
    private void servicePt(){}
    @Around("DataAdvice.servicePt()")
    // @Around("servicePt()")这两种写法都对
    public Object trimStr(ProceedingJoinPoint pjp) throws Throwable {
        //获取原始方法的参数
        Object[] args = pjp.getArgs();
        for (int i = 0; i < args.length; i++) {
            //判断参数是不是字符串
            if(args[i].getClass().equals(String.class)){
                args[i] = args[i].toString().trim();
            }
        }
        //将修改后的参数传入到原始方法的执行中
        Object ret = pjp.proceed(args);
        return ret;
    }
}
步骤5:运行程序


不管密码root前后是否加空格,最终控制台打印的都是true


步骤6:优化测试


为了能更好的看出AOP已经生效,我们可以修改ResourcesImpl类,在方法中将密码的长度进行打印

@Repository
public class ResourcesDaoImpl implements ResourcesDao {
    public boolean readResources(String url, String password) {
        System.out.println(password.length());
        //模拟校验
        return password.equals("root");
    }
}

再次运行成功,就可以根据最终打印的长度来看看,字符串的空格有没有被去除掉。


注意:


网络异常,图片无法展示
|


5,AOP总结


AOP的知识就已经讲解完了,接下来对于AOP的知识进行一个总结:


5.1 AOP的核心概念


  • 概念:AOP(Aspect Oriented Programming)面向切面编程,一种编程范式
  • 作用:在不惊动原始设计的基础上为方法进行功能增强
  • 核心概念
  • 代理(Proxy):SpringAOP的核心本质是采用代理模式实现的
  • 连接点(JoinPoint):在SpringAOP中,理解为任意方法的执行
  • 切入点(Pointcut):匹配连接点的式子,也是具有共性功能的方法描述
  • 通知(Advice):若干个方法的共性功能,在切入点处执行,最终体现为一个方法
  • 切面(Aspect):描述通知与切入点的对应关系
  • 目标对象(Target):被代理的原始对象成为目标对象


5.2 切入点表达式


  • 切入点表达式标准格式:动作关键字(访问修饰符  返回值  包名.类/接口名.方法名(参数)异常名)
execution(* com.itheima.service.*Service.*(..))


  • 切入点表达式描述通配符:
  • 作用:用于快速描述,范围描述
  • *:匹配任意符号(常用)
  • .. :匹配多个连续的任意符号(常用)
  • +:匹配子类类型
  • 切入点表达式书写技巧
    1.按标准规范开发
    2.查询操作的返回值建议使用*匹配
    3.减少使用..的形式描述包
    4.对接口进行描述,使用表示模块名,例如UserService的匹配描述为Service
    5.方法名书写保留动词,例如get,使用表示名词,例如getById匹配描述为getBy
    6.参数根据实际情况灵活调整


5.3 五种通知类型


  • 前置通知
  • 后置通知
  • 环绕通知(重点)
  • 环绕通知依赖形参ProceedingJoinPoint才能实现对原始方法的调用
  • 环绕通知可以隔离原始方法的调用执行
  • 环绕通知返回值设置为Object类型
  • 环绕通知中可以对原始方法调用过程中出现的异常进行处理
  • 返回后通知
  • 抛出异常后通知


5.4 通知中获取参数


  • 获取切入点方法的参数,所有的通知类型都可以获取参数
  • JoinPoint:适用于前置、后置、返回后、抛出异常后通知
  • ProceedingJoinPoint:适用于环绕通知
  • 获取切入点方法返回值,前置和抛出异常后通知是没有返回值,后置通知可有可无,所以不做研究
  • 返回后通知
  • 环绕通知
  • 获取切入点方法运行异常信息,前置和返回后通知是不会有,后置通知可有可无,所以不做研究
  • 抛出异常后通知
  • 环绕通知


6,AOP事务管理


6.1 Spring事务简介


6.1.1 相关概念介绍


  • 事务作用:在数据层保障一系列的数据库操作同成功同失败
  • Spring事务作用:在数据层或业务层保障一系列的数据库操作同成功同失败


数据层有事务我们可以理解,为什么业务层也需要处理事务呢?


举个简单的例子,


  • 转账业务会有两次数据层的调用,一次是加钱一次是减钱
  • 把事务放在数据层,加钱和减钱就有两个事务
  • 没办法保证加钱和减钱同时成功或者同时失败
  • 这个时候就需要将事务放在业务层进行处理。


Spring为了管理事务,提供了一个平台事务管理器PlatformTransactionManager


网络异常,图片无法展示
|


commit是用来提交事务,rollback是用来回滚事务。


PlatformTransactionManager只是一个接口,Spring还为其提供了一个具体的实现:


网络异常,图片无法展示
|


从名称上可以看出,我们只需要给它一个DataSource对象,它就可以帮你去在业务层管理事务。其内部采用的是JDBC的事务。所以说如果你持久层采用的是JDBC相关的技术,就可以采用这个事务管理器来管理你的事务。而Mybatis内部采用的就是JDBC的事务,所以后期我们Spring整合Mybatis就采用的这个DataSourceTransactionManager事务管理器。

目录
相关文章
|
XML Java 程序员
😧 Spring_day03(一)
😧 Spring_day03
105 1
|
Java 关系型数据库 MySQL
😧 SpringBoot(三)
😧 SpringBoot
103 1
|
druid Java 关系型数据库
😧 SpringBoot(四)
😧 SpringBoot
104 0
|
XML 前端开发 Java
😧 SpringBoot(二)
😧 SpringBoot
89 1
|
XML Java 关系型数据库
😧 Spring_day02 ✅(四)
😧 Spring_day02 ✅
66 0
|
XML Java Maven
😧 Spring_day02 ✅(二)
😧 Spring_day02 ✅
72 0
😧 Spring_day02 ✅(二)
|
存储 Java Maven
😧 Spring_day01 ✅(二)
😧 Spring_day01 ✅
75 0
|
Java 数据库连接 测试技术
😧 Spring_day03(六)
😧 Spring_day03
84 0
|
Java Maven Spring
😧 Spring_day01 ✅(三)
😧 Spring_day01 ✅
125 0
|
Java Spring 容器
😧 Spring_day03(二)
😧 Spring_day03
96 0