欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

自定义事务注解

程序员文章站 2024-01-20 15:45:16
...

自定义事务注解


大纲:

1.如何自定义注解
2.如何封装一个事务注解
步骤:1.spring.xml 开启注解
	2.定义注解接口
	3.写事务类
	4.写切面类
4.写切面类的步骤
1.先通过反射拿到代理对象的方法代理对象的信息
2.通过方法获取到这个方法是否加入了注解
    (JpcTc jpcTc = objMethod.getDeclaredAnnotation(JpcTc.class);)
3.如果有则开启事务
4.如果没有则不执行事务

1.写spring.xml

<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  xmlns:p="http://www.springframework.org/schema/p"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:aop="http://www.springframework.org/schema/aop"  xmlns:tx="http://www.springframework.org/schema/tx"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd">
       <context:component-scan base-package="com.cn"></context:component-scan>
       <aop:aspectj-autoproxy></aop:aspectj-autoproxy> <!-- 开启事物注解 -->
       <!-- 1. 数据源对象: C3P0连接池 -->
       <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
              <property name="driverClass"  value="com.mysql.jdbc.Driver"></property>
              <property name="jdbcUrl"  value="jdbc:mysql://127.0.0.1:3306/test"></property>
              <property name="user" value="root"></property>
              <property name="password" value="root"></property>
       </bean>
       <!-- 2. JdbcTemplate工具类实例 -->
       <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
              <property name="dataSource" ref="dataSource"></property>
       </bean>
       <!-- 3.配置事务 -->
       <bean id="dataSourceTransactionManager"
              class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
              <property name="dataSource" ref="dataSource"></property>
       </bean>
       <tx:annotation-driven transaction-manager="dataSourceTransactionManager" />
</beans>

2.事务工具类

package com.cn.util;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Scope;
import org.springframework.jdbc.datasource.DataSourceTransactionManager;
importorg.springframework.stereotype.Component;
import org.springframework.transaction.TransactionStatus;
import org.springframework.transaction.interceptor.DefaultTransactionAttribute;
@Component
@Scope("prototype") // 每个事务都是新的实例 目的解决线程安全问题 多例子  这个注解很重要!!!!  有点像给他分配 一个 threadLoad 防止多线程并发
public class Transaction {
       private TransactionStatus transactionStatus;
       @Autowired
       private DataSourceTransactionManager dataSourceTransactionManager;
       
       public TransactionStatus  begin(){
              TransactionStatus status =  dataSourceTransactionManager.getTransaction(new DefaultTransactionAttribute());
              return status;
       }
       
       //提交事务
       public void commit(TransactionStatus status){
              dataSourceTransactionManager.commit(status);
       }
       //回滚事务
       public void rollback(){
              dataSourceTransactionManager.rollback(transactionStatus);
       }
       
}

User类

package com.cn.service;
//user 服务层
public interface UserService {
       public void add();
       public void del();
}

User实现类

package com.cn.service.impl;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import com.cn.dao.UserDao;
import com.cn.inteface.JpcTc;
import com.cn.service.UserService;
//user 服务层
@Service
public class UserServiceImpl implements UserService {
       @Autowired
       private UserDao userDao;
       
       
       @JpcTc
       public void add() {
              // 调用接口的时候 接口失败 需要回滚,但是日志记录不需要回滚。
              userDao.add("test001", 20);
               int i = 1 / 0;
              System.out.println("################");
              userDao.add("test002", 21);
       }
       // 方法执行完毕之后,才会提交事务
       public void del() {
              System.out.println("del");
       }
}

注解类

package com.cn.inteface;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target(value = ElementType.METHOD) // 这个就是注意 他只能在方法上
@Retention(RetentionPolicy.RUNTIME)    // 必须要加这个注解 这个注解才能反射
public @interface JpcTc {
//这就定义了一个注解
}

dao层

package com.cn.dao;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.stereotype.Repository;

@Repository
public class UserDao {
    @Autowired
    private JdbcTemplate jdbcTemplate;

    public void add(String name, Integer age) {
        String sql = "INSERT INTO t_users(NAME, age) VALUES(?,?);";
        int updateResult = jdbcTemplate.update(sql, name, age);
        System.out.println("updateResult:" + updateResult);
    }

}

aop切面

package com.cn.aop;

import java.lang.reflect.Method;

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
import org.springframework.transaction.TransactionStatus;

import com.cn.inteface.JpcTc;
import com.cn.util.Transaction;

@Aspect
@Component
public class AopConfig {

    @Autowired
    private Transaction transaction ;
    
    @Pointcut("execution(* com.cn.service.*.*.*(..))")
    public void print(){}
    
    @AfterThrowing("execution(* com.cn.service.*.*.*(..))")
    public void afterThrowing() {
        // 获取当前事务进行回滚
        // TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
        System.out.println("抛异常,事务要回滚了");
        transaction.rollback();
    }

    
    @Around("print()")
    public void auout(ProceedingJoinPoint pjp) throws Throwable{
        /*
         * 1.先通过反射 获取到 方法  方法上是否存在注解
         * 2.如果有的话则开启事务
         * 3.执行代码
         * 4.判断是否有开启 事务, 如果有 就结束事务
         */
        String methodName = pjp.getSignature().getName();
        // 获取目标对象
        Class<?> classTarget = pjp.getTarget().getClass();
        // 获取目标对象类型
        Class<?>[] par = ((MethodSignature) pjp.getSignature()).getParameterTypes();
        // 获取目标对象方法
        Method objMethod = classTarget.getMethod(methodName, par);
        //这个方法就能 获取到  是否存在这个注解
        JpcTc jpcTc = objMethod.getDeclaredAnnotation(JpcTc.class);
        
        
        //###2.如果存在 则说明存在注解
        TransactionStatus status  = null;
        if(jpcTc != null){
            System.out.println("###########开启事务了!!!!!");
            status = transaction.begin();
        }
        //###3.执行被代理对象
        pjp.proceed();
        //###4.判断
        if(status != null){
            System.out.println("##########提交事务了!!!!!");
transaction.commit(status);
        }
        
        
        
        
        
        
        
        
        
    }
}

main方法测试

package com.cn;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import com.cn.service.UserService;
public class Test001 {
       public static void main(String[] args) {
              ClassPathXmlApplicationContext applicationContext = new  ClassPathXmlApplicationContext("spring.xml");
              UserService userService = (UserService)  applicationContext.getBean("userServiceImpl");
              userService.add();
       }
}

自定义事务注解

相关标签: spirng aop