1.什么是AOP

Aspect  Orientied   Programming的简称,即 面向(方面)切面编程 ,不改变一个组件源代码的情况下 可以对组件功能进行增强。

例如:servlet中的过滤器,继承,装饰者模式,代理模式,

JDK 的代理 必须有统一接口 目标类和代理类 兄弟关系
CGLIB 的代理 底层是继承 目标类和代理类 父子关系

2.AOP 中涉及到核心概念
   ###切面 Aspect 抽取共通业务逻辑到一个类中 每个共通业务逻辑就是一个切面方法
   定义了共通业务逻辑的类叫切面类 使用切面类创建的对象 叫切面对象
   连接点 JoinPoint 添加切面逻辑的一个位置 一般这个就是一个方法
   ###切点 Pointcut 一堆连接点的集合 后面会讲表达式的写法
   目标对象 Target 那个对象需要加共通业务逻辑
   代理对象 Proxy 被增强了的目标对象就是代理对象
   ###通知 Advice 时机 (目标方法执行之前 之后 执行前后 出异常)

切面 通知 切点 (把共通的业务逻辑 在合适的时间点 加入到对应的方法上)

6.切点表达式的写法
6.1 bean 名字限定表达式
    bean(容器中组件的id) 组件的id可以使用统配
6.2 类型限定表达式
   within(限定的类型) 限定的类型 是通过包名.类名
   within(com.xdl.service.XdlUserService) 对 XdlUserService 类型做限定
   XdlUserService中所有方法都切入共通逻辑
   within(com.xdl.service.*) 对service 包下 所有类型 都切入对应的逻辑
   within(com.xdl.*.*) xdl 直接子包下的类型 切入对应的逻辑
   within(com.xdl..*) xdl 包 以及子包下的类型 切入对应的逻辑
6.3 方法限定表达式
   execution(权限修饰 方法返回值类型 方法名(参数) throws 异常)
   返回值类型 方法名(参数) 是必须的
   execution(void user*()) 所有的返回void 方法名以user开头的无参的方法
   被切入逻辑

7.通知的五种类型
   <aop:before 前置通知 目标方法执行之前执行
   <aop:after 最终通知 目标方法执行之后 最终肯定会执行
   <aop:after-returning 后置通知 目标方法执行之后 如果目标方法出现异常 则不执行
   <aop:after-throwing 异常通知 目标方法执行出现异常 就会调用异常通知
   <aop:around 环绕通知 目标方法执行前后都会调用

8. 五种通知对应的标注
   <aop:before 前置通知 @Before
   <aop:after 最终通知 @After
   <aop:after-returning 后置通知 @AfterReturning
   <aop:after-throwing 异常通知 @AfterThrowing
   <aop:around 环绕通知 @Around

例子:

首先拷贝一个配置文件到类路径下

<?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:context="http://www.springframework.org/schema/context"
xmlns:jdbc="http://www.springframework.org/schema/jdbc" xmlns:jee="http://www.springframework.org/schema/jee"
xmlns:tx="http://www.springframework.org/schema/tx" xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:mvc="http://www.springframework.org/schema/mvc" xmlns:util="http://www.springframework.org/schema/util"
xmlns:jpa="http://www.springframework.org/schema/data/jpa"
xsi:schemaLocation="
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.1.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.1.xsd
http://www.springframework.org/schema/jdbc http://www.springframework.org/schema/jdbc/spring-jdbc-4.1.xsd
http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-4.1.xsd
http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.1.xsd
http://www.springframework.org/schema/data/jpa http://www.springframework.org/schema/data/jpa/spring-jpa-1.3.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.1.xsd
http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-4.1.xsd
http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util-4.1.xsd">
<!-- 组件扫描 -->
<context:component-scan base-package="cn.com"></context:component-scan>
<!-- 通过配置织入@Aspectj切面-->
<aop:aspectj-autoproxy proxy-target-class="true"></aop:aspectj-autoproxy> </beans>

好了,开始写代码:

首先添加一个model类;

import java.io.Serializable;

public class BankAccountModel implements Serializable {
private static final long serialVersionUID = 1L;
private int id;
private String acc_no;
private String acc_password;
private double acc_money; public BankAccountModel() {
super();
} public BankAccountModel(int id, String acc_no, String acc_password, double acc_money) {
super();
this.id = id;
this.acc_no = acc_no;
this.acc_password = acc_password;
this.acc_money = acc_money;
} public int getId() {
return id;
} public void setId(int id) {
this.id = id;
} public String getAcc_no() {
return acc_no;
} public void setAcc_no(String acc_no) {
this.acc_no = acc_no;
} public String getAcc_password() {
return acc_password;
} public void setAcc_password(String acc_password) {
this.acc_password = acc_password;
} public double getAcc_money() {
return acc_money;
} public void setAcc_money(double acc_money) {
this.acc_money = acc_money;
} public static long getSerialversionuid() {
return serialVersionUID;
} @Override
public String toString() {
return "BankAccountModel [id=" + id + ", acc_no=" + acc_no + ", acc_password=" + acc_password + ", acc_money="
+ acc_money + "]";
} }

来一个接口和一个实现类

public interface BankAccountDao {

    int addAccount(BankAccountModel ba);//添加

    BankAccountModel selectAccountByID(int id); //查询

}
@Repository("bankDao")//持久层标注
public class BankAccountDaoIMP implements BankAccountDao { @Override
public int addAccount(BankAccountModel ba) {
System.out.println("正在添加......");
return 0;
} @Override
public BankAccountModel selectAccountByID(int id) {
System.out.println("正在查询......");
return null;
} }

然后再写一个service层,(好吧,其实是可以不用写的,毕竟没有真正实现功能)

@Service("bankService")//spring中service层的标注1
public class BankAccountService {
@Autowired
private BankAccountDao dao; public int addAccount(BankAccountModel ba) {
return dao.addAccount(ba);
} public BankAccountModel selectAccountByID(int id) {
return dao.selectAccountByID(id);
} public void login(){ System.out.println("登录中....");
if(1==2){
throw new RuntimeException("登录方法出现异常!!");//这一部分未来测试专门让他出异常的,可以忽略哦
} } public void register(){
System.out.println("正在注册....");
}
}

最后可以来一个aspect类了

@Component//通用层标注
@Aspect //基于aop的标注
public class BankAccountAspect { /*
* @Before("bean(bankA*)") public void beforeUserLog(){ System.out.println(
* "#### ####"); }
*/ @Before("within(cn.com.service..*)")
public void beforeUserLog() {
System.out.println("#### ####");
} /* 异常的参数必须出现在最后面 */
@AfterThrowing(pointcut = "within(cn.com.service..*)", throwing = "e")
public void beforeAround(JoinPoint j, Exception e) {
System.out.println("正在记录异常信息..." + e.getMessage() + ":" + j.getSignature());
} /*统计方法执行时间*/
@Around("execution(void *())")
public Object countTime(ProceedingJoinPoint pjp) throws Throwable{//此处的方法必须返回为Object,否则环绕的目标方法中如果有返回值就会丢弃
long stat=System.currentTimeMillis();
//执行目标方法
Object proceed = pjp.proceed();//使用proceed()方法来执行目标方法 long end = System.currentTimeMillis();
System.out.println("方法执行的毫秒数是:" + (end-stat));
return proceed;
} } 测试代码:

public static void main(String[] args) {
            ApplicationContext app=new ClassPathXmlApplicationContext("applicationContext.xml");
            BankAccountService user=app.getBean("bankService",BankAccountService.class);
            System.out.println(user.addAccount(new BankAccountModel(12, "13", "12324", 123)));
            System.out.println(user.selectAccountByID(2));
            }

测试结果:

#### ####
    正在添加......
    0
    #### ####
    正在查询......
    null

 

基于标注的AOP面向切面编程的更多相关文章

  1. 基于SpringBoot AOP面向切面编程实现Redis分布式锁

    基于SpringBoot AOP面向切面编程实现Redis分布式锁 基于SpringBoot AOP面向切面编程实现Redis分布式锁 基于SpringBoot AOP面向切面编程实现Redis分布式 ...

  2. 从壹开始前后端分离【 .NET Core2.0 +Vue2.0 】框架之十 || AOP面向切面编程浅解析:简单日志记录 + 服务切面缓存

    代码已上传Github+Gitee,文末有地址 上回<从壹开始前后端分离[ .NET Core2.0 Api + Vue 2.0 + AOP + 分布式]框架之九 || 依赖注入IoC学习 + ...

  3. Spring:AOP面向切面编程

    AOP主要实现的目的是针对业务处理过程中的切面进行提取,它所面对的是处理过程中的某个步骤或阶段,以获得逻辑过程中各部分之间低耦合性的隔离效果. AOP是软件开发思想阶段性的产物,我们比较熟悉面向过程O ...

  4. 详细解读 Spring AOP 面向切面编程(二)

    本文是<详细解读 Spring AOP 面向切面编程(一)>的续集. 在上篇中,我们从写死代码,到使用代理:从编程式 Spring AOP 到声明式 Spring AOP.一切都朝着简单实 ...

  5. Z从壹开始前后端分离【 .NET Core2.0/3.0 +Vue2.0 】框架之十 || AOP面向切面编程浅解析:简单日志记录 + 服务切面缓存

    本文梯子 本文3.0版本文章 代码已上传Github+Gitee,文末有地址 大神反馈: 零.今天完成的深红色部分 一.AOP 之 实现日志记录(服务层) 1.定义服务接口与实现类 2.在API层中添 ...

  6. 03-spring框架—— AOP 面向切面编程

    3.1 动态代理 动态代理是指,程序在整个运行过程中根本就不存在目标类的代理类,目标对象的代理对象只是由代理生成工具(不是真实定义的类)在程序运行时由 JVM 根据反射等机制动态生成的.代理对象与目标 ...

  7. AOP(面向切面编程)大概了解一下

    前言 上一篇在聊MemoryCache的时候,用到了Autofac提供的拦截器进行面向切面编程,很明显能体会到其优势,既然涉及到了,那就趁热打铁,一起来探探面向切面编程. 正文 1. 概述 在软件业, ...

  8. 极简SpringBoot指南-Chapter05-SpringBoot中的AOP面向切面编程简介

    仓库地址 w4ngzhen/springboot-simple-guide: This is a project that guides SpringBoot users to get started ...

  9. AOP 面向切面编程, Attribute在项目中的应用

    一.AOP(面向切面编程)简介 在我们平时的开发中,我们一般都是面对对象编程,面向对象的特点是继承.多态和封装,我们的业务逻辑代码主要是写在这一个个的类中,但我们在实现业务的同时,难免也到多个重复的操 ...

随机推荐

  1. mysql--多表连接查询

    一.多表连接查询 SELECT 字段列表 FROM 表1 INNER|LEFT|RIGHT JOIN 表2 ON 表1.字段 = 表2.字段; 1.交叉连接:不适用任何匹配条件.生成笛卡尔积 sele ...

  2. 有效单词词广场——算法面试刷题5(for google),考察数学

    给定一个单词序列,检查它是否构成一个有效单词广场.一个有效的单词广场应满足以下条件:对于满足0≤k<max(numRows numColumns)的k,第k行和第k列对应的字符串应该相同,. 给 ...

  3. 【OCP 12c】最新CUUG OCP-071考试题库(62题)

    62.(13-17)choose the best answer: You need to list the employees in DEPARTMENT_ID 30 in a single row ...

  4. mysql 基础整合大全

    mysql  数据库操作: 创建数据库: create database db_sanguo charset utf8; 切进db_sanguo use db_sanguo 删除数据库: drop d ...

  5. [Swift实际操作]七、常见概念-(14)使用UIColor设置界面组件的颜色属性

    打开移动应用程序,不可避免的需要和颜色打交道.本文将为你演示颜色对象的使用. 首先导入需要使用到的界面工具框架 import UIKit 通过UIColor的属性,可以获得橙色.右侧的实时反馈区,显示 ...

  6. Apache环境修改.htaccess文件实现子目录强制HTTPS访问

    如果要在Apache环境下实现子目录强制HTTPS地址访问,该怎么实现呢?在此文章中将与大家一起分享如何在Apache环境下修改.htaccess文件来实现子目录强制HTTPS地址访问. 1.根目录域 ...

  7. 问题 I: 闪闪发光

    [提交] [状态] [命题人:外部导入] 题目描述 一所位于云南昆明的中医药本科院校--云南中医学院. 因为报考某专业的人数骤减,正面临着停招的危机. 其中有九名少女想到一条妙计——成为偶像, 只要她 ...

  8. 【性能测试】:JVM内存监控策略的方法,以及监控结果说明

    JVM内存监控主要在稳定性压测期间,监控应用服务器内存泄露等问题: [JVM远程监控设置] 1.打开WAS控制台:https://ip:port/ibm/console/login.do 2.进入路径 ...

  9. Maven 安装jar文件到本地repository

    Reference: https://maven.apache.org/general.html#importing-jars mvn install:install-file \ -Dfile=&l ...

  10. Unity3D实现随机播放背景音频

    1.先在第一人称下新建空白物体,命名“audio” 2.在audio中加入Audio Source 3.在第一人称组件里添加Audio Liistener和Audio脚本 4.脚本中添加代码 usin ...