- 代理模式 代理模式,可以为目标类添加额外功能。
- Spring 动态代理
- 定义目标类对象
- 定义额外功能,增强。实现Spring内置的接口
- 配置增强类
- 定义切入点
- 编织组装
- 增强类型
- 前置增强 MethodBeforeAdvice
- 后置增强 AfterReturningAdvice
- 异常增强 ThrowsAdvice
- 环绕增强 MethodInterceptor
- 切入点表达式
- execution(返回值类型 包名.类名.方法名(参数表)) execution(* com.bcl.service.*.*(..));
- args(参数表)
- within(全类名)
- @annotation(自定义注解)
- 事务的隔离级别
1 再谈web.xml
1.1 web.xml中标签的加载顺序
到目前为止web.xml中出现的标签:servlet
filter
listener
context-param
。
加载顺序,从前到后:
context-param 定义一对键值对数据,通常为listener使用 listener ContextLoaderListener:在web应用启动时执行,创建Spring工厂 filter Struts2Filter:过滤所有请求 servlet
1.2 Servlet的url-pattern的配置方式
Servlet的url-pattern有4种配置格式:
精确匹配 /book/showAllBooks 路径匹配 /* /book/* 后缀名匹配 *.action *.do *.jsp 缺省匹配 / 在上面3种都无法匹配的情况下,做默认匹配
优先级从高到低:
1 精确匹配 2 路径匹配 3后缀名匹配 4缺省匹配
注意点:
- 同是路径匹配,路径最长者优先
- 路径匹配和后缀名匹配不能混淆使用。/book/*.do 错误
- 任何时候,无论怎么配置,filter一定优先于servlet
2 Spring整合MyBatis
2.1 MyBatis项目开发步骤
- 搭建开发环境
- 新建项目
- 导入依赖 数据库驱动依赖:ojdbcx.jar mysql-connector-java.jar MyBatis相关的依赖 servlet jsp jstl的依赖 struts2的依赖 junit hutool druid
- 配置文件 工具类 jdbc.properties log4j.properties xxxMapper.xml mybatis-config.xml struts.xml
- 配置文件初始化 web.xml 配置Struts2Filter mybatis-config.xml 初始化配置
- 建表
- 实体
- mapper
- 接口
- 实现:XxxMapper.xml
- service
- 接口
- 实现: SqlSession sqlSession = MyBatisUtils.openSession(); XxxMapper mapper = sqlSession.getMapper(XxxMapper.class);
- test
- action jsp
- 集成测试
2.2 整合思路
image-20200603111758115
2.3 整合实战
准备工作,添加 mybatis-spring
代码语言:javascript复制<!-- 整合spring mybatis的依赖-->
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis-spring</artifactId>
<version>2.0.4</version>
</dependency>
初版配置:
代码语言:javascript复制<!-- 读取jdbc.properties-->
<context:property-placeholder location="classpath:jdbc.properties"/>
<!-- 0 定义一个连接池 -->
<bean id="druidDataSource" class="com.alibaba.druid.pool.DruidDataSource">
<property name="driverClassName" value="${jdbc.driverClassName}"/>
<property name="url" value="${jdbc.url}"/>
<property name="username" value="${jdbc.username}"/>
<property name="password" value="${jdbc.password}"/>
</bean>
<!-- 定义SqlSessionFactory-->
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="druidDataSource"/>
<!--
配置实体类的包名,自动为实体配置短类名的别名
-->
<property name="typeAliasesPackage" value="com.bcl.entity"/>
<property name="mapperLocations">
<!-- 配置mapper.xml的路径-->
<list>
<!--<value>classpath:com/bcl/mapper/UserMapper.xml</value>
<value>classpath:com/bcl/mapper/StudentMapper.xml</value>
<value>classpath:com/bcl/mapper/BookMapper.xml</value>-->
<value>classpath:com/bcl/mapper/*Mapper.xml</value>
</list>
</property>
</bean>
<!-- 定义SqlSession -->
<bean id="sqlSession" class="org.mybatis.spring.SqlSessionTemplate">
<constructor-arg ref="sqlSessionFactory"/>
</bean>
<!-- 创建 UserMapper实现类对象-->
<bean id="userMapper" class="org.mybatis.spring.mapper.MapperFactoryBean">
<property name="sqlSessionTemplate" ref="sqlSession"/>
<property name="mapperInterface" value="com.bcl.mapper.UserMapper"/>
</bean>
<!--<bean id="bookMapper" class="org.mybatis.spring.mapper.MapperFactoryBean">
<property name="sqlSessionTemplate" ref="sqlSession"/>
<property name="mapperInterface" value="com.bcl.mapper.BookMapper"/>
</bean>-->
<bean id="userService" class="com.bcl.service.impl.UserServiceImpl">
<property name="userMapper" ref="userMapper"/>
</bean>
最终配置:
代码语言:javascript复制<!-- 读取jdbc.properties-->
<context:property-placeholder location="classpath:jdbc.properties"/>
<!-- 0 定义一个连接池 -->
<bean id="druidDataSource" class="com.alibaba.druid.pool.DruidDataSource">
<property name="driverClassName" value="${jdbc.driverClassName}"/>
<property name="url" value="${jdbc.url}"/>
<property name="username" value="${jdbc.username}"/>
<property name="password" value="${jdbc.password}"/>
</bean>
<!-- 定义SqlSessionFactory-->
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="druidDataSource"/>
<!--
配置实体类的包名,自动为实体配置短类名的别名
-->
<property name="typeAliasesPackage" value="com.bcl.entity"/>
<property name="mapperLocations">
<!-- 配置mapper.xml的路径-->
<list> <value>classpath:com/bcl/mapper/*Mapper.xml</value>
</list>
</property>
</bean>
<!--
自动创建Mapper实现类对象
自动扫描basePackage包下的Mapper接口,自动创建Mapper接口的实现类对象
-->
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
<!--
mapper实现类对象的id规则:接口名首字母小写
UserMapper ==> userMapper
BookMapper ==> bookMapper
-->
<property name="basePackage" value="com.bcl.mapper"/>
</bean>
<bean id="userService" class="com.bcl.service.impl.UserServiceImpl">
<property name="userMapper" ref="userMapper"/>
</bean>
2.4 Spring整合MyBatis项目的开发步骤
- 搭建开发环境
- 新建web项目
导入依赖
jdbc驱动依赖
mybatis依赖
struts2依赖
spring依赖
spring整合mybaits依赖
spring整合struts2依赖
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-test</artifactId>
<version>4.3.26.RELEASE</version>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.48</version>
</dependency>
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis</artifactId>
<version>3.5.4</version>
</dependency>
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-log4j12</artifactId>
<version>1.7.30</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>4.3.26.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aop</artifactId>
<version>4.3.26.RELEASE</version>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.5</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-web</artifactId>
<version>4.3.26.RELEASE</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>javax.servlet-api</artifactId>
<version>3.1.0</version>
</dependency>
<dependency>
<groupId>javax.servlet.jsp</groupId>
<artifactId>javax.servlet.jsp-api</artifactId>
<version>2.3.3</version>
</dependency>
<dependency>
<groupId>jstl</groupId>
<artifactId>jstl</artifactId>
<version>1.2</version>
</dependency>
<dependency>
<groupId>org.apache.struts</groupId>
<artifactId>struts2-core</artifactId>
<version>2.3.16.3</version>
</dependency>
<!-- 整合spring mybatis的依赖-->
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis-spring</artifactId>
<version>2.0.4</version>
</dependency>
<!-- druid的依赖-->
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>druid</artifactId>
<version>1.1.14</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-jdbc</artifactId>
<version>4.3.26.RELEASE</version>
</dependency>
<dependency>
<groupId>org.apache.struts</groupId>
<artifactId>struts2-spring-plugin</artifactId>
<version>2.3.16.3</version>
</dependency>
代码语言:javascript复制配置文件初始化
web.xml
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:applicationContext.xml</param-value>
</context-param>
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<filter>
<filter-name>Struts2Filter</filter-name>
<filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>Struts2Filter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
applicationContext.xml
<!-- 读取jdbc.properties-->
<context:property-placeholder location="classpath:jdbc.properties"/>
<!-- 0 定义一个连接池 -->
<bean id="druidDataSource" class="com.alibaba.druid.pool.DruidDataSource">
<property name="driverClassName" value="${jdbc.driverClassName}"/>
<property name="url" value="${jdbc.url}"/>
<property name="username" value="${jdbc.username}"/>
<property name="password" value="${jdbc.password}"/>
</bean>
<!-- 定义SqlSessionFactory-->
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="druidDataSource"/>
<!--
配置实体类的包名,自动为实体配置短类名的别名
-->
<property name="typeAliasesPackage" value="com.bcl.entity"/>
<property name="mapperLocations">
<!-- 配置mapper.xml的路径-->
<list>
<value>classpath:com/bcl/mapper/*Mapper.xml</value>
</list>
</property>
</bean>
<!--
自动创建Mapper实现类对象
自动扫描basePackage包下的Mapper接口,自动创建Mapper接口的实现类对象
-->
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
<!--
mapper实现类对象的id规则:接口名首字母小写
UserMapper ==> userMapper
BookMapper ==> bookMapper
-->
<property name="basePackage" value="com.bcl.mapper"/>
</bean>
- 建表
- 实体
- mapper
- service
- 接口
- 实现
class
XxxServiceImpl
implements
XxxService{
private XxxMapper xxxMapper;
public
void
setXxxMapper(XxxMapper mapper){
this.xxxMapper = mapper;
}
业务方法...
}
- test
- action jsp
XxxAction{
private XxxService xxxService;
public
void
setXxxService(XxxService service){
this.xxxService = service;
}
服务方法...
}
- 集成测试
3 Spring中事务控制
Spring提供2种控制方式:
- 编程式事务控制:在程序中定义事务控制代码。
- 声明式事务控制:借助Spring AOP实现,将事务控制的代码定义成功能增强,将增强编织到切点指定的位置(业务层)。
Spring AOP声明式事务控制的步骤:
- 定义原始类对象(service对象)
<!-- 定义目标类对象-->
<bean
id="userService"
class="com.bcl.service.impl.UserServiceImpl">
<property
name="userMapper"
ref="userMapper"/>
</bean>
- 定义增强类,Spring内置有事务控制的增强 DataSourceTransactionManager事务增强类
- 配置增强类
<!-- 配置增强类:事务管理器-->
<bean
id="txManager"
class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property
name="dataSource"
ref="druidDataSource"/>
</bean>
<tx:advice
transaction-manager="txManager"
id="txAdvice">
<tx:attributes>
<!-- login 开启只读事务-->
<tx:method
name="login"
read-only="true"/>
<!-- 其它的方法,都必须开启事务 -->
<tx:method
name="*"
propagation="REQUIRED"/>
</tx:attributes>
</tx:advice>
- 定义切点
- 编织组装
<!--
定义切入点
编织组装
-->
<aop:config>
<aop:pointcut
id="servicePointCut"
expression="execution(* com.bcl.service.*.*(..))"/>
<aop:advisor
advice-ref="txAdvice"
pointcut-ref="servicePointCut"/>
</aop:config>
4 事务详解
- read-only="true" 只读事务,不能执行增删改,但对于查询有性能优化。
- timeout 超时机制 如果超过执行时间,事务自动回滚。默认值-1,表示跟数据库的配置有关。
- rollback-for和no-rollback-for rollback-for 定义异常类型:事务遇到这种异常会回滚 no-rollback-for 定义异常类型:事务遇到这种异常不会回滚 Spring中,rollback-for的默认值是RuntimeException,no-rollback-for的默认值是Exception。 自定义异常时,发生该异常需要回滚事务,继承RuntimeException 自定义异常时,发生该异常不需要回滚事务,继承Exception
- propagation(传播机制) 企业开发时,业务复杂时,会出现业务方法调用业务方法的情况。propagation定义了一个业务方法被另外一个业务方法调用时,事务的传播方式。 常见的事务传播机制: REQUIRED 如果外部有事务,加入外部事务,如果没有则新建 SUPPORTS 如果外部有事务,加入外部事务,如果没有则以无事务的状态运行 REQUIRES_NEW 无论是否有外部事务,都会新建一个事务
- isolation(隔离级别) READ_UNCOMMITTED 读未提交 READ_COMMITTED 读提交 实战时使用 REPEATABLE_READ 可重复读 SERIALIZABLE 序列化读