先了解什么是注解

注解

Jdk1.5新增新技术,注解。很多框架为了简化代码,都会提供有些注解。可以理解为插件,是代码级别的插件,在类的方法上写:@XXX,就是在代码上插入了一个插件。

注解不会也不能影响代码的实际逻辑,仅仅起到辅助性的作用。

注解分类:内置注解(也成为元注解 jdk 自带注解)自定义注解(Spring框架

什么内置注解

(1) @SuppressWarnings   再程序前面加上可以在javac编译中去除警告--阶段是SOURCE
(2) @Deprecated   带有标记的包,方法,字段说明其过时----阶段是SOURCE
(3)@Overricle   打上这个标记说明该方法是将父类的方法重写--阶段是SOURCE

实现自定义注解

元注解的作用就是负责注解其他注解。Java5.0定义了4个标准的meta-annotation类型,它们被用来提供对其它 annotation类型作说明。Java5.0定义的元注解:
@Target

@Target说明了Annotation所修饰的对象范围:Annotation可被用于 packages、types(类、接口、枚举、Annotation类型)、类型成员(方法、构造方法、成员变量、枚举值)、方法参数和本地变量(如循环变量、catch参数)。在Annotation类型的声明中使用了target可更加明晰其修饰的目标。

  1. CONSTRUCTOR:用于描述构造器
  2. FIELD:用于描述域
  3. LOCAL_VARIABLE:用于描述局部变量
  4. METHOD:用于描述方法
  5. PACKAGE:用于描述包
  6. PARAMETER:用于描述参数
  7. TYPE:用于描述类、接口(包括注解类型) 或enum声明

2.@Retention

表示需要在什么级别保存该注释信息,用于描述注解的生命周期(即:被描述的注解在什么范围内有效)
3.@Documented
4.@interface

@Target(value = { ElementType.METHOD })
@Retention(RetentionPolicy.RUNTIME)
public @interface AddAnnotation {int userId() default 0;String userName() default "默认名称";String[]arrays();
}
反射读取注解信息public static void main(String[] args) throws ClassNotFoundException {Class classInfo = Class.forName("com.entity.User");// 获取到所有方法Method[] methods = classInfo.getDeclaredMethods();for (Method method : methods) {System.out.println(method);        //查看该方法是否存在注解AddAnnotation declaredAnnotation = method.getDeclaredAnnotation(AddAnnotation.class);if (declaredAnnotation == null) {// 结束本次循环continue;}// 获取userIdint userId = declaredAnnotation.userId();System.out.println("userId:" + userId);// 获取userNameString userName = declaredAnnotation.userName();System.out.println("userName:" + userName);// 获取arraysString[] arrays = declaredAnnotation.arrays();for (String str : arrays) {System.out.println("str:" + str);}}}

自定义事务注解

//编程事务(需要手动begin 手动回滚  手都提交)
@Component()
@Scope("prototype") // 设置成原型解决线程安全
public class TransactionUtils {private TransactionStatus transactionStatus;// 获取事务源
    @Autowiredprivate DataSourceTransactionManager dataSourceTransactionManager;// 开启事务public TransactionStatus begin() {transactionStatus = dataSourceTransactionManager.getTransaction(new DefaultTransactionAttribute());return transactionStatus;}// 提交事务public void commit(TransactionStatus transaction) {dataSourceTransactionManager.commit(transaction);}// 回滚事务public void rollback() {System.out.println("rollback");dataSourceTransactionManager.rollback(transactionStatus);}}注解类@Autowiredprivate TransactionUtils transactionUtils;@AfterThrowing("execution(* com.itmayiedu.service.*.*.*(..))")public void afterThrowing() throws NoSuchMethodException, SecurityException {// isRollback(proceedingJoinPoint);System.out.println("程序发生异常");// TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();// TransactionStatus currentTransactionStatus =// TransactionAspectSupport.currentTransactionStatus();// System.out.println("currentTransactionStatus:" +// currentTransactionStatus);
        transactionUtils.rollback();}// // 环绕通知 在方法之前和之后处理事情@Around("execution(* com.itmayiedu.service.*.*.*(..))")public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {// 调用方法之前执行TransactionStatus transactionStatus = begin(proceedingJoinPoint);proceedingJoinPoint.proceed();// 代理调用方法 注意点: 如果调用方法抛出异常不会执行后面代码// 调用方法之后执行
        commit(transactionStatus);}public TransactionStatus begin(ProceedingJoinPoint pjp) throws NoSuchMethodException, SecurityException {// // 判断是否有自定义事务注解ExtTransaction declaredAnnotation = getExtTransaction(pjp);if (declaredAnnotation == null) {return null;}// 如果有自定义事务注解,开启事务System.out.println("开启事务");TransactionStatus transactionStatu = transactionUtils.begin();return transactionStatu;}public void commit(TransactionStatus transactionStatu) {if (transactionStatu != null) {// 提交事务System.out.println("提交事务");transactionUtils.commit(transactionStatu);}}public ExtTransaction getExtTransaction(ProceedingJoinPoint pjp) throws NoSuchMethodException, SecurityException {// 获取方法名称String methodName = pjp.getSignature().getName();// 获取目标对象Class<?> classTarget = pjp.getTarget().getClass();// 获取目标对象类型Class<?>[] par = ((MethodSignature) pjp.getSignature()).getParameterTypes();// 获取目标对象方法Method objMethod = classTarget.getMethod(methodName, par);// // 判断是否有自定义事务注解ExtTransaction declaredAnnotation = objMethod.getDeclaredAnnotation(ExtTransaction.class);if (declaredAnnotation == null) {System.out.println("您的方法上,没有加入注解!");return null;}return declaredAnnotation;}// 回滚事务public void isRollback(ProceedingJoinPoint pjp) throws NoSuchMethodException, SecurityException {// // 判断是否有自定义事务注解ExtTransaction declaredAnnotation = getExtTransaction(pjp);if (declaredAnnotation != null) {System.out.println("已经开始回滚事务");// 获取当前事务 直接回滚
            TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();return;}}使用自定义注解@ExtTransaction
public void add() {
userDao.add("test001", 20);
int i = 1 / 0;
System.out.println("################");
userDao.add("test002", 21);
}

Spring事物传播行为

Spring中事务的定义:

Propagation(key属性确定代理应该给哪个方法增加事务行为。这样的属性最重要的部份是传播行为。)有以下选项可供使用:

PROPAGATION_REQUIRED—如果当前有事务,就用当前事务,如果当前没有事务,就新建一个事务。这是最常见的选择。

PROPAGATION_SUPPORTS--支持当前事务,如果当前没有事务,就以非事务方式执行。//如果外层方法没有事务,就会以非事务进行执行。

PROPAGATION_MANDATORY--支持当前事务,如果当前没有事务,就抛出异常。

PROPAGATION_REQUIRES_NEW--新建事务,如果当前存在事务,把当前事务挂起。

PROPAGATION_NOT_SUPPORTED--以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。

--- 如果当前有事务,就是以非事务进行执行

PROPAGATION_NEVER--以非事务方式执行,如果当前存在事务,则抛出异常。

默认传播行为为REQUIRED

转载于:https://www.cnblogs.com/a1304908180/p/10615436.html

《四 spring源码》spring的事务注解@Transactional 原理分析相关推荐

  1. Spring源码解析之-- 事务TransactionInterceptor 分析(开启事务)

    目录 一.介绍 二.TransactionInterceptor 分析 2. 流程 2.1 invoke 2.1.1 TransactionAspectSupport#invokeWithinTran ...

  2. 4、Eureka 源码解析 之 Eureka Client 启动原理分析

    在前面的一篇文章 3.Eureka 源码解析 之 Eureka Server 启动原理分析当中我们分析了一下 Eureka Server 的启动.在集群环境下 Eureka Server 相互之前需要 ...

  3. spring系列-注解驱动原理及源码-声明式事务使用及原理解析

    目录 一.环境准备 1.JdbcTemplate使用实例 2.事务添加 二.声明式事务源码分析 1.原理(与AOP非常相似) 一.环境准备 1.JdbcTemplate使用实例 (1)pom文件添加依 ...

  4. Spring源码——声明式事务流程

    前言 最近回顾了一下Spring源码,准备用思维导图的方式简单的将整个源码内容的流程展示出来,思维导图.图片等文件更新在https://github.com/MrSorrow/spring-frame ...

  5. Spring源码解析之@Component注解的扫描

    阅读须知 Spring源码版本:4.3.8 文章中使用/* */注释的方法会做深入分析 正文 承接Spring源码解析之context:component-scan标签解析,下面就是扫描的流程: Cl ...

  6. Spring源码系列(十三)——Spring源码编译及详细注解

    文章目录 1. 环境搭建 2. 代码编译 2.1 编译代码 2.1.1 build.gradle 2.1.1.1 第一处 2.1.1.2 第二处 2.1.2 gradle.properties 2.1 ...

  7. Spring源码——Spring MVC

    前言 内容主要参考自<Spring源码深度解析>一书,算是读书笔记或是原书的补充.进入正文后可能会引来各种不适,毕竟阅读源码是件极其痛苦的事情. 本文主要涉及书中第十一章的部分,依照书中内 ...

  8. Spring源码解析 - AbstractBeanFactory 实现接口与父类分析

    2019独角兽企业重金招聘Python工程师标准>>> 我们先来看类图吧: 除了BeanFactory这一支的接口,AbstractBeanFactory主要实现了AliasRegi ...

  9. 死磕Spring源码-常见底层核心注解

    一:Spring框架功能整体介绍 二:根据Spring架构图可知,包含以上的模块儿.下面逐个模块儿进行解释. 1:IOC 和 DI(Inversion of Control I DependencyI ...

  10. c++ map 获取key列表_好未来Golang源码系列一:Map实现原理分析

    分享老师:学而思网校 郭雨田 一.map的结构与设计原理 golang中map是一个kv对集合.底层使用hash table,用链表来解决冲突 ,出现冲突时,不是每一个key都申请一个结构通过链表串起 ...

最新文章

  1. 结构体名和结构体名是个指针的区别
  2. 外部中断0——51程序
  3. boost::polygon模块voronoi相关的测试程序
  4. Delta Lake在Soul的应用实践
  5. JS循环执行函数setInterval
  6. 活动子项父项的复杂CSS选择器[重复]
  7. HDOJ水题集合1:最小生成树(Kruskal)
  8. 深入解析 Flink 的算子链机制
  9. 信号与系统 拉普拉斯变换
  10. mrpoid模拟器java版_mrpoid模拟器2019下载
  11. 前端实现antd文本域限制字数
  12. 获取选股宝7x24小时数据
  13. Power Query介绍
  14. PS 修改图片上的文字内容但保留文字格式
  15. 【photoshop】笔记(四)之修补工具
  16. 小学文化学导数——斜率
  17. appium java模拟微信登录,python实战之结合Appium自动化操作微信
  18. TL431工作原理、经典应用电路、输出产生真的的原因分析
  19. (每日一读2019.10.17)基于SE(2)-XYZ约束的地面车辆视觉里程计定位与地图构建
  20. python调用matlab需要的库_python调用matlab文件

热门文章

  1. matlab gui怎样将结果保存在excel中_声发射简单使用matlab导入参数波形数据并绘图...
  2. 牛客网 ACM模式单行输入输出规范
  3. 老生常谈exec函数族
  4. 【UVA1339】古老的密码(巧妙思路+(q)sort降序排列的三种方法)
  5. hdoj1290切球形蛋糕(递推和划分问题)
  6. epoll监听文件_epoll
  7. python输入一个序列_Python序列合并,python
  8. android桌面adw,ADW Launcher
  9. 接口测试及服务器性能压测,接口测试及服务器性能压测
  10. frame边框阴影html,CSS阴影效果的比较之drop-Shadow与box-Shadow