Spring框架的第三天


课程回顾:Spring框架第二天

1. IOC的注解方式
	* @Value
	* @Resource(name="")

2. 整合JUnti单元测试
	* 先有Junit4运行环境
	* 导入spring-test.jar包

3. Spring的AOP技术(XML的方式)
	* 什么是AOP:面向切面编程
	* 采用代理技术完成(默认采用JDK接口方式,CGLIB技术代理对象)
	* AOP的入门
		* 编写切面类(编写通知的方法)
		* 配置AOP
		* 切入点的表达式(execution([public] * 包名.类名.方法(..)))

今天内容

1. Spring框架的AOP之注解的方式
2. Spring框架的JDBC模板
3. Spring框架的事务管理

案例一:使用Spring框架的AOP技术对DAO层的功能进行增强


案例一:使用Spring框架的AOP技术对DAO层的功能进行增强

1. 使用Spring框架的AOP技术对DAO层的功能进行增强

技术分析之:Spring框架的AOP技术(注解方式)

1. 步骤一:创建JavaWEB项目,引入具体的开发的jar包
	* 先引入Spring框架开发的基本开发包
	* 再引入Spring框架的AOP的开发包
		* spring的传统AOP的开发的包
			* spring-aop-4.2.4.RELEASE.jar
			* com.springsource.org.aopalliance-1.0.0.jar

		* aspectJ的开发包
			* com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar
			* spring-aspects-4.2.4.RELEASE.jar

2. 步骤二:创建Spring的配置文件,引入具体的AOP的schema约束
	<beans xmlns="http://www.springframework.org/schema/beans"
	       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	       xmlns:aop="http://www.springframework.org/schema/aop"
	       xsi:schemaLocation="
			http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
			http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">

	</beans>

3. 步骤三:创建包结构,编写具体的接口和实现类
	* com.itheima.demo1
		* CustomerDao			-- 接口
		* CustomerDaoImpl		-- 实现类

4. 步骤四:将目标类配置到Spring中
	<bean id="customerDao" class="com.itheima.demo1.CustomerDaoImpl"/>

5. 步骤五:定义切面类
	* 添加切面和通知的注解
		* @Aspect					-- 定义切面类的注解

		* 通知类型(注解的参数是切入点的表达式)
			* @Before				-- 前置通知
		    * @AfterReturing		-- 后置通知
		    * @Around				-- 环绕通知
		    * @After				-- 最终通知
		    * @AfterThrowing		-- 异常抛出通知

	* 具体的代码如下
		@Aspect
		public class MyAspectAnno {
			@Before(value="execution(public void com.itheima.demo1.CustomerDaoImpl.save())")
			public void log(){
				System.out.println("记录日志...");
			}
		}

6. 步骤六:在配置文件中定义切面类
	<bean id="myAspectAnno" class="com.itheima.demo1.MyAspectAnno"/>

7. 步骤七:在配置文件中开启自动代理
	<aop:aspectj-autoproxy/>

8. 完成测试
	@RunWith(SpringJUnit4ClassRunner.class)
	@ContextConfiguration("classpath:applicationContext.xml")
	public class Demo1 {

		@Resource(name="customerDao")
		private CustomerDao customerDao;

		@Test
		public void run1(){
			customerDao.save();
			customerDao.update();
		}
	}

技术分析之通知类型

1. 通知类型
	* @Before				-- 前置通知
    * @AfterReturing		-- 后置通知
    * @Around				-- 环绕通知(目标对象方法默认不执行的,需要手动执行)
    * @After				-- 最终通知
    * @AfterThrowing		-- 异常抛出通知

2. 配置通用的切入点
	* 使用@Pointcut定义通用的切入点

	@Aspect
	public class MyAspectAnno {
		@Before(value="MyAspectAnno.fn()")
		public void log(){
			System.out.println("记录日志...");
		}
		@Pointcut(value="execution(public void com.itheima.demo1.CustomerDaoImpl.save())")
		public void fn(){}
	}

案例二:Spring框架的事务管理完成转账的案例


需求分析

1. 完成一个转账的功能,需要进行事务的管理,使用Spring的事务管理的方式完成

Spring框架的JDBC模板技术


技术分析之Spring框架的JDBC模板技术概述

1. Spring框架中提供了很多持久层的模板类来简化编程,使用模板类编写程序会变的简单
2. 提供了JDBC模板,Spring框架提供的
	* JdbcTemplate类

3. Spring框架可以整合Hibernate框架,也提供了模板类
	* HibernateTemplate类

技术分析之演示JDBC的模板类

1. 步骤一:创建数据库的表结构
	create database spring_day03;
	use spring_day03;
	create table t_account(
		id int primary key auto_increment,
		name varchar(20),
		money double
	);

2. 引入开发的jar包
	* 先引入IOC基本的6个jar包
	* 再引入Spring-aop的jar包
	* 最后引入JDBC模板需要的jar包
		* MySQL数据库的驱动包
		* Spring-jdbc.jar
		* Spring-tx.jar

3. 编写测试代码(自己来new对象的方式)
	@Test
	public void run1(){
		// 创建连接池,先使用Spring框架内置的连接池
		DriverManagerDataSource dataSource = new DriverManagerDataSource();
		dataSource.setDriverClassName("com.mysql.jdbc.Driver");
		dataSource.setUrl("jdbc:mysql:///spring_day03");
		dataSource.setUsername("root");
		dataSource.setPassword("root");
		// 创建模板类
		JdbcTemplate jdbcTemplate = new JdbcTemplate(dataSource);
		// 完成数据的添加
		jdbcTemplate.update("insert into t_account values (null,?,?)", "测试",10000);
	}

技术分析之使用Spring框架来管理模板类

1. 刚才编写的代码使用的是new的方式,应该把这些类交给Spring框架来管理。
2. 修改的步骤如下
	* 步骤一:Spring管理内置的连接池
		<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
	    	<property name="driverClassName" value="com.mysql.jdbc.Driver"/>
	    	<property name="url" value="jdbc:mysql:///spring_day03"/>
	    	<property name="username" value="root"/>
	    	<property name="password" value="root"/>
	    </bean>

	* 步骤二:Spring管理模板类
		<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
	    	<property name="dataSource" ref="dataSource"/>
	    </bean>

	* 步骤三:编写测试程序
		@RunWith(SpringJUnit4ClassRunner.class)
		@ContextConfiguration("classpath:applicationContext.xml")
		public class Demo2 {

			@Resource(name="jdbcTemplate")
			private JdbcTemplate jdbcTemplate;

			@Test
			public void run2(){
				jdbcTemplate.update("insert into t_account values (null,?,?)", "测试2",10000);
			}
		}

技术分析之Spring框架管理开源的连接池

1. 管理DBCP连接池
	* 先引入DBCP的2个jar包
		* com.springsource.org.apache.commons.dbcp-1.2.2.osgi.jar
		* com.springsource.org.apache.commons.pool-1.5.3.jar

	* 编写配置文件
		<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource">
	    	<property name="driverClassName" value="com.mysql.jdbc.Driver"/>
	    	<property name="url" value="jdbc:mysql:///spring_day03"/>
	    	<property name="username" value="root"/>
	    	<property name="password" value="root"/>
	    </bean>

2. 管理C3P0连接池
	* 先引入C3P0的jar包
		* com.springsource.com.mchange.v2.c3p0-0.9.1.2.jar

	* 编写配置文件
		<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
	    	<property name="driverClass" value="com.mysql.jdbc.Driver"/>
	    	<property name="jdbcUrl" value="jdbc:mysql:///spring_day03"/>
	    	<property name="user" value="root"/>
	    	<property name="password" value="root"/>
	    </bean>

技术分析之Spring框架的JDBC模板的简单操作

1. 增删改查的操作
	@RunWith(SpringJUnit4ClassRunner.class)
	@ContextConfiguration("classpath:applicationContext.xml")
	public class SpringDemo3 {

		@Resource(name="jdbcTemplate")
		private JdbcTemplate jdbcTemplate;

		@Test
		// 插入操作
		public void demo1(){
			jdbcTemplate.update("insert into account values (null,?,?)", "冠希",10000d);
		}

		@Test
		// 修改操作
		public void demo2(){
			jdbcTemplate.update("update account set name=?,money =? where id = ?", "思雨",10000d,5);
		}

		@Test
		// 删除操作
		public void demo3(){
			jdbcTemplate.update("delete from account where id = ?", 5);
		}

		@Test
		// 查询一条记录
		public void demo4(){
			Account account = jdbcTemplate.queryForObject("select * from account where id = ?", new BeanMapper(), 1);
			System.out.println(account);
		}

		@Test
		// 查询所有记录
		public void demo5(){
			List<Account> list = jdbcTemplate.query("select * from t_account", new BeanMapper());
			for (Account account : list) {
				System.out.println(account);
			}
		}
	}

	class BeanMapper implements RowMapper<Account>{
		public Account mapRow(ResultSet rs, int arg1) throws SQLException {
			Account account = new Account();
			account.setId(rs.getInt("id"));
			account.setName(rs.getString("name"));
			account.setMoney(rs.getDouble("money"));
			return account;
		}
	}

技术分析之Spring框架的事务管理


技术分析之事务的回顾

1. 事务:指的是逻辑上一组操作,组成这个事务的各个执行单元,要么一起成功,要么一起失败!
2. 事务的特性
	* 原子性
	* 一致性
	* 隔离性
	* 持久性

3. 如果不考虑隔离性,引发安全性问题
	* 读问题:
		* 脏读:
		* 不可重复读:
		* 虚读:

	* 写问题:
		* 丢失更新:

4. 如何解决安全性问题
	* 读问题解决,设置数据库隔离级别
	* 写问题解决可以使用 悲观锁和乐观锁的方式解决

技术分析之Spring框架的事务管理相关的类和API

1. PlatformTransactionManager接口		-- 平台事务管理器.(真正管理事务的类)。该接口有具体的实现类,根据不同的持久层框架,需要选择不同的实现类!
2. TransactionDefinition接口			-- 事务定义信息.(事务的隔离级别,传播行为,超时,只读)
3. TransactionStatus接口				-- 事务的状态

4. 总结:上述对象之间的关系:平台事务管理器真正管理事务对象.根据事务定义的信息TransactionDefinition 进行事务管理,在管理事务中产生一些状态.将状态记录到TransactionStatus中

5. PlatformTransactionManager接口中实现类和常用的方法
	1. 接口的实现类
		* 如果使用的Spring的JDBC模板或者MyBatis框架,需要选择DataSourceTransactionManager实现类
		* 如果使用的是Hibernate的框架,需要选择HibernateTransactionManager实现类

	2. 该接口的常用方法
		* void commit(TransactionStatus status)
		* TransactionStatus getTransaction(TransactionDefinition definition)
		* void rollback(TransactionStatus status)

6. TransactionDefinition
	1. 事务隔离级别的常量
		* static int ISOLATION_DEFAULT 					-- 采用数据库的默认隔离级别
		* static int ISOLATION_READ_UNCOMMITTED
		* static int ISOLATION_READ_COMMITTED
		* static int ISOLATION_REPEATABLE_READ
		* static int ISOLATION_SERIALIZABLE

	2. 事务的传播行为常量(不用设置,使用默认值)
		* 先解释什么是事务的传播行为:解决的是业务层之间的方法调用!!

		* PROPAGATION_REQUIRED(默认值)	-- A中有事务,使用A中的事务.如果没有,B就会开启一个新的事务,将A包含进来.(保证A,B在同一个事务中),默认值!!
		* PROPAGATION_SUPPORTS			-- A中有事务,使用A中的事务.如果A中没有事务.那么B也不使用事务.
		* PROPAGATION_MANDATORY			-- A中有事务,使用A中的事务.如果A没有事务.抛出异常.

		* PROPAGATION_REQUIRES_NEW(记)-- A中有事务,将A中的事务挂起.B创建一个新的事务.(保证A,B没有在一个事务中)
		* PROPAGATION_NOT_SUPPORTED		-- A中有事务,将A中的事务挂起.
		* PROPAGATION_NEVER 			-- A中有事务,抛出异常.

		* PROPAGATION_NESTED(记)		-- 嵌套事务.当A执行之后,就会在这个位置设置一个保存点.如果B没有问题.执行通过.如果B出现异常,运行客户根据需求回滚(选择回滚到保存点或者是最初始状态)

技术分析之搭建事务管理转账案例的环境(强调:简化开发,以后DAO可以继承JdbcDaoSupport类)

1. 步骤一:创建WEB工程,引入需要的jar包
	* IOC的6个包
	* AOP的4个包
	* C3P0的1个包
	* MySQL的驱动包
	* JDBC目标2个包
	* 整合JUnit测试包

2. 步骤二:引入配置文件
	* 引入配置文件
		* 引入log4j.properties

		* 引入applicationContext.xml
			<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
		    	<property name="driverClass" value="com.mysql.jdbc.Driver"/>
		    	<property name="jdbcUrl" value="jdbc:mysql:///spring_day03"/>
		    	<property name="user" value="root"/>
		    	<property name="password" value="root"/>
		    </bean>

3. 步骤三:创建对应的包结构和类
	* com.itheima.demo1
		* AccountService
		* AccountServlceImpl
		* AccountDao
		* AccountDaoImpl

4. 步骤四:引入Spring的配置文件,将类配置到Spring中
	<bean id="accountService" class="com.itheima.demo1.AccountServiceImpl">
	</bean>

	<bean id="accountDao" class="com.itheima.demo1.AccountDaoImpl">
	</bean>

5. 步骤五:在业务层注入DAO ,在DAO中注入JDBC模板(强调:简化开发,以后DAO可以继承JdbcDaoSupport类)
	<bean id="accountService" class="com.itheima.demo1.AccountServiceImpl">
		<property name="accountDao" ref="accountDao"/>
	</bean>

	<bean id="accountDao" class="com.itheima.demo1.AccountDaoImpl">
		<property name="dataSource" ref="dataSource"/>
	</bean>

6. 步骤六:编写DAO和Service中的方法
	public class AccountDaoImpl extends JdbcDaoSupport implements AccountDao {
		public void outMoney(String out, double money) {
			this.getJdbcTemplate().update("update t_account set money = money = ? where name = ?", money,out);
		}
		public void inMoney(String in, double money) {
			this.getJdbcTemplate().update("update t_account set money = money + ? where name = ?", money,in);
		}
	}

7. 步骤七:编写测试程序.
	@RunWith(SpringJUnit4ClassRunner.class)
	@ContextConfiguration("classpath:applicationContext.xml")
	public class Demo1 {

		@Resource(name="accountService")
		private AccountService accountService;

		@Test
		public void run1(){
			accountService.pay("冠希", "美美", 1000);
		}
	}


技术分析之Spring框架的事务管理的分类

1. Spring的事务管理的分类
	1. Spring的编程式事务管理(不推荐使用)
		* 通过手动编写代码的方式完成事务的管理(不推荐)

	2. Spring的声明式事务管理(底层采用AOP的技术)
		* 通过一段配置的方式完成事务的管理(重点掌握注解的方式)

技术分析之Spring框架的事务管理之编程式的事务管理(了解)

1. 说明:Spring为了简化事务管理的代码:提供了模板类 TransactionTemplate,所以手动编程的方式来管理事务,只需要使用该模板类即可!!

2. 手动编程方式的具体步骤如下:
	1. 步骤一:配置一个事务管理器,Spring使用PlatformTransactionManager接口来管理事务,所以咱们需要使用到他的实现类!!
		<!-- 配置事务管理器 -->
		<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
			<property name="dataSource" ref="dataSource"/>
		</bean>

	2. 步骤二:配置事务管理的模板
		<!-- 配置事务管理的模板 -->
		<bean id="transactionTemplate" class="org.springframework.transaction.support.TransactionTemplate">
			<property name="transactionManager" ref="transactionManager"/>
		</bean>

	3. 步骤三:在需要进行事务管理的类中,注入事务管理的模板.
		<bean id="accountService" class="com.itheima.demo1.AccountServiceImpl">
			<property name="accountDao" ref="accountDao"/>
			<property name="transactionTemplate" ref="transactionTemplate"/>
		</bean>

	4. 步骤四:在业务层使用模板管理事务:
		// 注入事务模板对象
		private TransactionTemplate transactionTemplate;
		public void setTransactionTemplate(TransactionTemplate transactionTemplate) {
			this.transactionTemplate = transactionTemplate;
		}

		public void pay(final String out, final String in, final double money) {
			transactionTemplate.execute(new TransactionCallbackWithoutResult() {

				protected void doInTransactionWithoutResult(TransactionStatus status) {
					// 扣钱
					accountDao.outMoney(out, money);
					int a = 10/0;
					// 加钱
					accountDao.inMoney(in, money);
				}
			});
		}


Spring框架的事务管理之声明式事务管理,即通过配置文件来完成事务管理(AOP思想)

1. 声明式事务管理又分成两种方式
	* 基于AspectJ的XML方式(重点掌握)
	* 基于AspectJ的注解方式(重点掌握)

Spring框架的事务管理之基于AspectJ的XML方式(重点掌握)

1. 步骤一:恢复转账开发环境

2. 步骤二:引入AOP的开发包

3. 步骤三:配置事务管理器
	<!-- 配置事务管理器 -->
	<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
		<property name="dataSource" ref="dataSource"/>
	</bean>

4. 步骤四:配置事务增强
	<!-- 配置事务增强 -->
	<tx:advice id="txAdvice" transaction-manager="transactionManager">
		<tx:attributes>
			<!--
				name		:绑定事务的方法名,可以使用通配符,可以配置多个。
				propagation	:传播行为
				isolation	:隔离级别
				read-only	:是否只读
				timeout		:超时信息
				rollback-for:发生哪些异常回滚.
				no-rollback-for:发生哪些异常不回滚.
			 -->
			<!-- 哪些方法加事务 -->
			<tx:method name="pay" propagation="REQUIRED"/>
		</tx:attributes>
	</tx:advice>

5. 步骤五:配置AOP的切面
	<!-- 配置AOP切面产生代理 -->
	<aop:config>
    	<aop:advisor advice-ref="myAdvice" pointcut="execution(* com.itheima.demo2.AccountServiceImpl.pay(..))"/>
    </aop:config>

	* 注意:如果是自己编写的切面,使用<aop:aspect>标签,如果是系统制作的,使用<aop:advisor>标签。

6. 步骤六:编写测试类
	@RunWith(SpringJUnit4ClassRunner.class)
	@ContextConfiguration("classpath:applicationContext2.xml")
	public class Demo2 {

		@Resource(name="accountService")
		private AccountService accountService;

		@Test
		public void run1(){
			accountService.pay("冠希", "美美", 1000);
		}
	}

Spring框架的事务管理之基于AspectJ的注解方式(重点掌握,最简单的方式)

1. 步骤一:恢复转账的开发环境

2. 步骤二:配置事务管理器
	<!-- 配置事务管理器  -->
	<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
		<property name="dataSource" ref="dataSource"/>
	</bean>

3. 步骤三:开启注解事务
	<!-- 开启注解事务 -->
	<tx:annotation-driven transaction-manager="transactionManager"/>

4. 步骤四:在业务层上添加一个注解:@Transactional

5. 编写测试类
	@RunWith(SpringJUnit4ClassRunner.class)
	@ContextConfiguration("classpath:applicationContext3.xml")
	public class Demo3 {

		@Resource(name="accountService")
		private AccountService accountService;

		@Test
		public void run1(){
			accountService.pay("冠希", "美美", 1000);
		}
	}

Eclipse需要做设置

1. 统一工作空间的编码,选择UTF-8
2. 把创建JSP页面的编码修改UTF-8
3. 重新配置Tomcat服务器
	* 先配置Tomcat服务器
	* 选择服务器 --> open --> 选择发布项目的目录(webapps目录)

4. SSH自己配置约束
10-05 16:28