首先,AOP和OOP类似,都是一种编程思想,OOP便是常说的面向对象编程,AOP则是面向切面编程,OOP的执行结构是纵向的,而AOP是横向的。直白点儿说,OOP的代码执行过程是按照顺序一步一步往下执行,而AOP则是将这些看似流程式的执行过程切断,在断层处加入更多的逻辑代码,比如,web后台代码的执行过程是从controller层 —> service层 —> DAO层,而我们通常会把从controller层到service层的执行流切断,形成一个切面,从而在这些切面上添加事务操作。其实这种操作不仅限于对事务操作的扩展,我们更要学会的是这种编程思想。
接下来,我们就以spring中配置事务的过程,详细介绍一下spring中的AOP编程
下面是我简单搭建起来的一个web工程,只有一个注册功能,接下来的代码分析就会按照这个工程的顺序往下介绍
首先我们的请求来自于controller层,那么我将controller层的代码粘贴如下:
package cn.minghao.controller;
import cn.minghao.pojo.User;
import cn.minghao.service.UserService;
import me.shib.java.lib.utils.JsonUtil;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.*;
import org.springframework.web.client.HttpClientErrorException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
@Controller
@RequestMapping("usr")
public class UserController {
private JsonUtil jsonUtil = new JsonUtil();
@Autowired
private UserService userService;
@RequestMapping(value = "/register", method = RequestMethod.POST, produces = "application/json;charset=UTF-8")
@ResponseBody
public String register(@RequestBody String jsonParam, HttpServletResponse response) {
try {
User user = new JsonUtil().fromJson(jsonParam, User.class);
userService.addUser(user);
response.setStatus(200);
return jsonUtil.toJson("OK");
} catch (HttpClientErrorException ce) {
response.setStatus(ce.getStatusCode().value());
//TODO
ce.printStackTrace();
return jsonUtil.toJson(ce.getMessage());
} catch (Exception e) {
response.setStatus(500);
//TODO
e.printStackTrace();
return jsonUtil.toJson(e.getMessage());
}
}
}
这个controller中只有一个register方法,实现的功能就是用户注册的功能,controller中调用了service层的addUser方法,而我们的addUser方法是数据库的更新操作,所以必须为其配置事务。
第一步,切点配置,为了能够使Spring扫描到addUser方法,所以首先我们配置AOP的切入点,此处再解释一个概念——切面,AOP的切面=切入点+通知,如果我们不配置任何通知的话,切入点就等于切面了,在本例中我们即将要配置的通知便是事务管理器,那么通知就是我们切断一个顺序执行流后,想在断层处新增的处理逻辑,此处的通知即是事务处理逻辑。
<aop:config>
<!-- 定义切点,所有的service的所有方法 -->
<aop:pointcut id="txPointcut" expression="execution(* cn.minghao.service.*.*(..))"/>
</aop:config>
第二步,数据源配置,切点配好后我们应当给切点的方法上加入事务操作形成完整切面,但是事务需要依赖数据源,所以首先我们配置一个数据源,数据源其实就是一个数据库连接池,事务管理器能从数据源中获得数据库连接
<!-- 数据源 -->
<bean id="dataSource" class="com.zaxxer.hikari.HikariDataSource">
<property name="driverClassName" value="${jdbc.driver}"/>
<property name="jdbcUrl" value="${jdbc.url}"/>
<property name="username" value="${jdbc.username}"/>
<property name="password" value="${jdbc.password}"/>
<property name="maximumPoolSize" value="2" />
<property name="minimumIdle" value="1" />
</bean>
第三步,配置事务管理器,将数据源赋值给事务管理器
<!-- 定义事务管理器 -->
<bean id="transa