spring事务管理方式,aop

时间:2023-12-28 21:52:44

达内12 note unit 09 01

1.spring事务管理

2.spring提供了对事务管理支持

spring采用aop机制完成事务控制

可以实现在不修改原有组件代码情况下实现事务控制功能。

spring提供了两种事务管理方式:

a。编程式事务管理(编写java代码)

  TransactionTemplate

b.声明式事务管理(编写配置,大家都用这种)

  xml版本配置

  注解版本配置

  --配置DataSourceTransactionManager

  --开启事务注解配置<tx:annotation>

  --在目标组件方法前添加@Transactional

注解版本例子:

例如UserServiceImpl中的regist注册方法需要事务:

第一步,我们现在applicationContext.xml中配置事务管理组件

<!--  配置事务管理组件 -->

<bean id="txManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">

  <property name="dataSource" ref="dbcp"></property>

</bean>

<!--  开启事务注解标记@Transactional -->

<!--  当调用带@Transactional 标记的方法时,将txManager事务管理功能切入到方法-->

<tx:annotation-driven transaction-manager="txManager"/>

第二步,在所有service类上,加上@Transactional注解

package org.alexhe.note.service;

import javax.annotation.Resource;

import org.alexhe.note.dao.IUserDao;

import org.alexhe.note.entity.NoteResult;

import org.alexhe.note.entity.User;

import org.alexhe.note.util.NoteUtil;

import org.springframework.stereotype.Service;

@Service("userService")

@Transactional//这里加入事务注解

public class UserServiceImpl implements IUserService{

@Resource

private IUserDao userDao;//注入

@Override

public NoteResult checkLogin(String name, String pwd) throws Exception {

// TODO Auto-generated method stub

NoteResult result=new NoteResult();

User user=userDao.findByName(name);

if(user==null){

result.setStatus(1);

result.setMsg("用户名不存在");

return result;

}

String md5_pwd=NoteUtil.md5(pwd);

if(!user.getCn_user_password().equals(md5_pwd)){

result.setStatus(2);

result.setMsg("密码不正确");

return result;

}

result.setStatus(0);

result.setMsg("用户名和密码正确");

result.setData(user.getCn_user_id());//返回userid

return result;

}

@Override

public NoteResult regist(String name, String password, String nickname) throws Exception {

NoteResult result=new NoteResult();

//检测用户名是否被占用

User has_user=userDao.findByName(name);

if(has_user!=null){

result.setStatus(1);

result.setMsg("用户名已被占用");

return result;

}

//注册

User user=new User();

user.setCn_user_name(name);

user.setCn_user_desc(nickname);

String md5_pwd=NoteUtil.md5(password);

user.setCn_user_password(md5_pwd);//设置加密的密码

String userId=NoteUtil.createId();

user.setCn_user_id(userId);//设置userid

//调用userDao保存

userDao.save(user);

result.setStatus(0);

result.setMsg("注册成功");

return result;

}

}

xml版本配置例子:(配置比注解版复杂)

第一步,我们现在applicationContext.xml中配置事务管理组件,记得xml里加入aop的头

<!--  配置事务管理组件 -->

<bean id="txManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">

  <property name="dataSource" ref="dbcp"></property>

</bean>

<tx:advice id="txAdvice" transaction-manager="txManager">

  <tx:attributes><!--  哪些方法用事务,就写在里面 -->

    <tx:method name="regist"/>

    <tx:method name="checkLogin"/>

    <tx:method name="add*"/> <!--  以add开头的所有方法 -->

   <!--  <tx:method name="*"/> <!--  所有方法都加注释 -->

  </tx:attributes>

</tx:advice>

<aop:config>

  <aop:pointcut id="target" expression="within(org.alexhe.note.service..*)"/> <!--  expression代表哪个组件,作用在哪些组件上,这里代表service包及其下面的所有组件-->

  <aop:advisor advice-ref="txAdvice" pointcut-ref="target"/>

</aop:config>

3.Spring对事务管理的控制

a。控制事务可读可写特性

Spring分为可读写事务和只读事务。默认为可读写,一般只涉及查询操作,建议用只读事务

@Transactional(readOnly=true)

b.控制事务是否回滚

Spring遇到runtimeException异常,会回滚。遇到非运行时异常,不会回滚。

@Transactional(readOnly=true,rollbackFor=IOException.class)   这样遇到IOException也会发生回滚。

建议:自定义异常继承自RuntimeException继承

public class MyException extends RuntimeException

c。控制事务传播类型

遇到带有事务控制方法调用另一个事务控制方法时,可以选择合适的传播类型,默认是required类型,后者使用前者事务。

d。控制事务隔离级别

@Transactional(readOnly=true, isolation=Isolation.READ_COMMITED);

由低到高如下:

READ_UNCOMMITED读未提交

READ_COMMITED读已提交

REPEATABLE_READ可重复读

SERIALIZABLE序列化操作

DEFAULT默认,根据数据库隔离级别自动选择,

4.spring aop应用

aop编程优点:可以动态将一个组件功能切入到指定的目标方法上。可以使结构更加灵活,也能实现组件重复利用。

aop编程:更注重于业务逻辑隔离,将一些共通处理逻辑和传统处理逻辑解耦。

例如事务处理,日志记录,异常处理等等。

适用环境:

  --共通处理逻辑

  --调用时机相同

例子,用xml配置方式,往controller上加方法:

1.新建了一个aspect包(非必须)

2.包里新建一个类,然后新建一个clogger方法。表示加上日志功能。

3.spring的文件里加配置,把需要aop的controller加上第二步类里的方法

package org.alexhe.note.aspect;

public class NoteLogger {

public void clogger(){

System.out.println("进入Controller处理");

}

}

spring的配置:

<!-- aop示例 -->

<bean id="noteLogger" class="org.alexhe.note.aspect.NoteLogger"></bean>

<aop:config>

<!-- 把上面的noteLogger定义为切面组件 -->

<aop:aspect ref="noteLogger">

<!-- 什么时候,向哪些方法上切入 --><!-- 在controler包及其子包下,所有执行的方法前,加入clogger方法  -->

<aop:before method="clogger" pointcut="within(org.alexhe.note.controller..*)"/>

</aop:aspect>

</aop:config>

例子,用注解方式,往service层加方法:

spring配置不用上面这一坨,只要加上:

<!-- 开启aop注解支持,@Aspect,@通知标记 -->

<aop:aspectj-autoproxy />

java类:

package org.alexhe.note.aspect;

import org.aspectj.lang.annotation.Aspect;

import org.aspectj.lang.annotation.Before;

import org.springframework.stereotype.Component;

@Component//扫描,将组件扫描到Spring容器

@Aspect//将当前组件设置为切面组件

public class ServiceLogger {

@Before("within(org.alexhe.note.service..*)")//service下面所有方法加入这个slooger方法

public  void slogger(){

System.out.println("进入service方法");

}

}

=====AOP应用=====

a。要切入什么功能

b。要切入的时机,什么时候切入,通知。

  前置通知,在原有方法前插入新功能。@Before

  后置通知,在执行完原有方法后,调入新的切面方法。@AfterReturning

  异常通知,在原有方法出异常了,调入新的切面方法。@AfterThrowing

  最终通知,不管有没有异常,最终都要走他。@After

  环绕通知=前置+后置

try{

  前置通知@Before

  //目标方法处理

  后置通知@AfterReturning

}catch(){

  异常通知@AfterThrowing

}finally{

  最终通知@After

}

c。往那些组件方法切入-->切入点

  --类型限定表达式

  within(类型)

  与类型匹配的组件都是目标

  within(org.service.UserService)

  within(org.service.*) 仅限于当前包下

  within(org.service..*) 当前包和子包下

  --方法限定表达式

  execution(修饰符 返回类型 方法名(参数) 抛出异常)     返回类型和方法名参数是必须的,其他可以省略

  execution(* find*(..))     必须是find开头的方法,参数返回值不限制

  execution(* org.service.UserService.regist*(..))

  execution(* org.service..*.*(..))

上述表达式可以使用 !,&&,|| 运算符连接。

案例:将异常信息写入文件

package org.alexhe.note.aspect;

import org.aspectj.lang.annotation.Aspect;

import org.aspectj.lang.annotation.AfterThrowing;

import org.springframework.stereotype.Component;

@Component//扫描,将组件扫描到Spring容器

@Aspect//将当前组件设置为切面组件

public class ExceptionLogger {

@AfterThrowing(throwing="e",pointcut="within(org.alexhe.note.service..*)")//service下面所有方法加入这个log方法

public  void log(Exception e){

System.out.println(e);

}

}