spring


LOGO

 

Author:Aaron

Version:9.0.2

 

一、引言1.1 原生web开发中存在哪些问题?二、Spring框架2.1 概念2.2 访问与下载三、Spring架构组成四、自定义工厂4.1 配置文件4.2 工厂类五、构建Maven项目5.1 新建项目5.2 选择Maven目录5.3 GAV坐标六、Spring环境搭建6.1 pom.xml中引入Spring常用依赖6.2 创建Spring配置文件七、Spring工厂编码八、依赖与配置文件详解8.1 Spring依赖关系8.2 schema九、IOC(Inversion of Control )控制反转【重点9.1 项目中强耦合问题9.2 解决方案十、DI(Dependency Injection)依赖注入【重点10.1 概念10.2 Set注入10.2.1 定义目标Bean类型10.2.2 基本类型 + 字符串类型 + 日期类型10.2.3 容器类型10.2.4 自建类型10.3 构造注入【了解】10.3.1 定义目标Bean类型10.3.2 注入10.4 自动注入【了解】十一、Bean细节11.1 控制简单对象的单例、多例(原型)模式11.2 FactoryBean创建复杂对象【了解】11.2.1 实现FactoryBean接口11.2.2 配置spring-context.xml11.2.3 特例十二、Spring工厂特性12.1 饿汉式创建优势12.2 生命周期方法12.3 生命周期注解 12.4 生命周期阶段十三、代理设计模式13.1 概念13.2 静态代理设计模式 13.3 动态代理设计模式13.3.1 JDK动态代理实现(基于接口)13.3.2 CGlib动态代理实现(基于继承)十四、面向切面编程【重点14.1 概念14.2 作用14.3 AOP开发术语14.4 环境搭建14.5 开发流程14.6 AOP小结14.7 通知类【可选】 14.8 通配切入点14.9 JDK和CGLIB选择14.10 后处理器14.10.1 后处理器定义14.10.2 配置后处理器14.10.3 bean生命周期十五、Spring + MyBatis【重点15.1 配置数据源15.1.1 创建数据库和表 15.1.2 整合Spring配置文件和properties配置文件15.1.3 dao接口和pojo实体15.1.4 service接口和实现类15.1.5 Druid监控中心(了解)15.1.6 测试监控中心(了解)15.2 整合MyBatis15.2.1 配置SqlSessionFactory15.2.2 配置MapperScannerConfigurer15.2.3 配置Service15.2.4 测试十六、声明式事务【重点16.1 事务属性16.1.1 隔离级别16.1.1.1 概念16.1.1.2 特性16.1.1.3 并发问题16.1.2 传播行为16.1.3 读写性16.1.4 事务超时16.1.5 事务回滚16.2 配置DataSourceTransactionManager16.3 配置事务通知16.4 编织16.5 测试十七、注解开发17.1 开启注解17.2 声明bean17.3 注入(DI)17.4 注解所需配置17.5 事务控制17.6 AOP开发17.6.1 配置17.6.2 注解使用十八、集成JUnit18.1 导入依赖18.2 编码

 

一、引言


1.1 原生web开发中存在哪些问题?

  • 传统Web开发存在硬编码所造成的过度程序耦合(例如:Service中作为属性的Dao对象)。

  • 部分Java EE API较为复杂,使用效率低(例如:JDBC开发步骤)。

  • 侵入性强,移植性差(例如:DAO实现的更换,从Connection到SqlSession)。

 

二、Spring框架


2.1 概念

  • Spring是一个项目管理框架,同时也是一套Java EE解决方案。

  • Spring是众多优秀设计模式的组合(工厂、单例、代理、适配器、包装器、观察者、模板、策略)。

  • Spring并未替代现有框架产品,而是将众多框架进行有机整合,简化企业级开发,俗称"胶水框架"。

 

2.2 访问与下载

官方网站:https://spring.io/

下载地址:http://repo.spring.io/release/org/springframework/spring/

 

三、Spring架构组成


Spring架构由诸多模块组成,可分类为

  • 核心技术:依赖注入,事件,资源,i18n,验证,数据绑定,类型转换,SpEL,AOP

  • 测试:模拟对象,TestContext框架,Spring MVC测试,WebTestClient。

  • 数据访问:事务,DAO支持,JDBC,ORM,封送XML。

  • Spring MVC和 Spring WebFlux Web框架。

  • 集成:远程处理,JMS,JCA,JMX,电子邮件,任务,调度,缓存。

  • 语言:Kotlin,Groovy,动态语言。

Spring架构组成
001
GroupIdArtifactId说明
org.springframework spring-beans Beans 支持,包含 Groovy
org.springframework spring-aop 基于代理的AOP支持
org.springframework spring-aspects 基于AspectJ 的切面
org.springframework spring-context 应用上下文运行时,包括调度和远程抽象
org.springframework spring-context-support 支持将常见的第三方类库集成到 Spring 应用上下文
org.springframework spring-core 其他模块所依赖的核心模块
org.springframework spring-expression Spring 表达式语言,SpEL
org.springframework spring-instrument JVM 引导的仪表(监测器)代理
org.springframework spring-instrument-tomcat Tomcat 的仪表(监测器)代理
org.springframework spring-jdbc 支持包括数据源设置和 JDBC 访问支持
org.springframework spring-jms 支持包括发送/接收JMS消息的助手类
org.springframework spring-messaging 对消息架构和协议的支持
org.springframework spring-orm 对象/关系映射,包括对 JPA 和 Hibernate 的支持
org.springframework spring-oxm 对象/XML 映射(Object/XML Mapping,OXM)
org.springframework spring-test 单元测试和集成测试支持组件
org.springframework spring-tx 事务基础组件,包括对 DAO 的支持及 JCA 的集成
org.springframework spring-web web支持包,包括客户端及web远程调用
org.springframework spring-webmvc REST web 服务及 web 应用的 MVC 实现
org.springframework spring-webmvc-portlet 用于 Portlet 环境的MVC实现
org.springframework spring-websocket WebSocket 和 SockJS 实现,包括对 STOMP 的支持
org.springframework spring-jcl Jakarta Commons Logging 日志系统

四、自定义工厂


4.1 配置文件

项目resources目录下创建bean.properties文件

userDao=com.qf.dao.UserDAOImpl
userService=com.qf.service.UserServiceImpl

4.2 工厂类

/**
* 自定义工厂
*/
public class MyFactory {
   private Properties properties = new Properties();
   
   public MyFactory(){}
   
   public MyFactory(String config) throws IOException {
       // 加载配置文件
       properties.load(MyFactory.class.getResourceAsStream(config));
  }
   
   // 获取对象
   public Object getBean(String beanName) throws ClassNotFoundException, IllegalAccessException, InstantiationException {
       // 获得类路径
       String classPath = properties.getProperty(beanName);
       if(classPath!=null){
           Class claz = null;
           // 反射:加载类对象
           claz = Class.forName(classPath);
           // 反射:获得对象
           return claz.newInstance();
      }
       return null;
  }
   
   public static void main(String[] args) throws Exception {

       // 创建工厂对象
       MyFactory myFactory = new MyFactory("/bean.properties");
       // 从工厂中获取对象
       UserDao userDao = (UserDao)myFactory.getbean("userDao");
       UserService userService = (UserService)myFactory.getbean("userService");
       userDao.selectUserAll();
       userService.findUserAll();
  }
}

 

五、构建Maven项目


5.1 新建项目

使用IDEA打开已创建的文件夹目录
002

 

5.2 选择Maven目录

选择Maven项目
003

 

5.3 GAV坐标

GAV坐标
004

 

六、Spring环境搭建


6.1 pom.xml中引入Spring常用依赖

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
        xsi:schemaLocation=
        "http://maven.apache.org/POM/4.0.0
         http://maven.apache.org/xsd/maven-4.0.0.xsd">
 
   <modelVersion>4.0.0</modelVersion>

   <groupId>com.qf</groupId>
   <artifactId>hello-spring</artifactId>
   <version>1.0-SNAPSHOT</version>

   <dependencies>
       <!-- Spring常用依赖 -->
       <dependency>
           <groupId>org.springframework</groupId>
           <artifactId>spring-context</artifactId>
           <version>5.1.6.RELEASE</version>
       </dependency>
       
       <dependency>
           <groupId>junit</groupId>
           <artifactId>junit</artifactId>
           <version>4.12</version>
       </dependency>
   </dependencies>
</project>

 

6.2 创建Spring配置文件

命名无限制,约定俗成命名有:spring-context.xml、applicationContext.xml、beans.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
      xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
      xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
 
</beans>

七、Spring工厂编码


定义目标Bean类型

public class MyClass{
   public void show(){
       System.out.println("HelloWorld");
  }
}

spring-context.xml中的< beans >内部配置bean标签

<!-- 配置实例(id:“唯一标识”  class="需要被创建的目标对象全限定名") -->
<bean id="mc" class="com.qf.spring.part1.factory.MyClass" />

调用Spring工厂API(ApplicationContext接口)

public class TestFactory{
   /**
    * 程序中的对象都交由Spring的ApplicationContext工厂进行创建。
    */
   public static void main(String[] args){
       //1. 读取配置文件中所需创建的bean对象,并获得工厂对象
       ApplicationContext ctx = new ClassPathXmlApplicationContext("spring-context.xml");
       //2. 通过id获取bean对象
MyClass mc = (MyClass) ctx.getBean("mc");
       //3. 使用对象
mc.show();
  }
}

八、依赖与配置文件详解


Spring框架包含多个模块,每个模块各司其职,可结合需求引入相关依赖Jar包实现功能。

8.1 Spring依赖关系

Spring常用功能的Jar包依赖关系
image-20191230164517693

 

8.2 schema

配置文件中的顶级标签中包含了语义化标签的相关信息

  • xmlns:语义化标签所在的命名空间。

  • xmlns:xsi:XMLSchema-instance 标签遵循Schema标签标准。

  • xsi:schemaLocation:xsd文件位置,用以描述标签语义、属性、取值范围等。

九、IOC(Inversion of Control )控制反转【重点


Inverse Of Controll:控制反转

反转了依赖关系的满足方式,由之前的自己创建依赖对象,变为由工厂推送。(变主动为被动,即反转)

解决了具有依赖关系的组件之间的强耦合,使得项目形态更加稳健

 

IOC是什么? IOC(Inversion of Control)控制反转,IOC是一种新的Java编程模式,目前很多轻量级容器都在广泛使用的模式。

IOC解决了什么问题? 在IOC出现以前,组件之间的协调关系是由程序内部代码来控制的,或者说,以前我们使用New关键字来实现两组间之间的依赖关系的。这种方式就造成了组件之间的互相耦合。IOC(控制反转)就是来解决这个问题的,它将实现组件间的关系从程序内部提到外部容器来管理。也就是说,由容器在运行期将组件间的某种依赖关系动态的注入组件中。

 

9.1 项目中强耦合问题

public class UserDAOImpl implements UserDAO{....}
public class UserServiceImpl implements UserService {
   // !!!强耦合了UserDAOImpl!!!,使得UserServiceImpl变得不稳健!!
   private UserDAO userDAO= new UserDAOImpl();
   @Override
   public User queryUser() {
       return userDAO.queryUser();
  }
  ....
}

9.2 解决方案

// 不引用任何一个具体的组件(实现类),在需要其他组件的位置预留存取值入口(set/get)
public class UserServiceImpl implements UserService {
   // !!!不再耦合任何DAO实现!!!,消除不稳健因素!!
   private UserDAO userDAO;
   // 为userDAO定义set/get,允许userDAO属性接收spring赋值
   //Getters And Setters
   @Override
   public User queryUser() {
       return userDAO.queryUser();
  }
  ....
}
<bean id="userDAO" class="com.qf.spring.part1.injection.UserDaoImpl"></bean>
<!-- UserServiceImpl组件 -->
<bean id="userService" class="com.qf.spring.part1.injection.UserServiceImpl">
   <!-- 由spring为userDAO属性赋值,值为id="userDAO"的bean -->
   <property name="userDAO" ref="userDAO"/>
</bean>

此时,如果需要更换其他UserDAO实现类,则UserServiceImpl不用任何改动!

则此时的UserServiceImpl组件变得更加稳健!

十、DI(Dependency Injection)依赖注入【重点


IoC的实现方式 及 与DI的关系?

  1. 依赖查找(Dependency Lookup):容器中的受控对象通过容器的API来查找自己所依赖的资源和协作对象。 这种方式虽然降低了对象间的依赖,但是同时也使用到了容器的API,造成了我们无法在容器外使用和测试对象。 依赖查找是一种更加传统的IoC实现方式。

  2. 依赖注入(Dependency Injection):这就是DI,字面上理解,依赖注入就是将服务注入到使用它的地方。对象只提供普通的方法让容器去决定依赖关系, 容器全权负责组件的装配,它会把符合依赖关系的对象通过属性(JavaBean中的setter)或者是构造传递给需要的对象。 相对于IoC而言,依赖注入(DI)更加准确地描述了IoC的设计理念。所谓依赖注入,即组件之间的依赖关系由容器在应用系统运行期来决定, 也就是由容器动态地将某种依赖关系的目标对象实例注入到应用系统中的各个关联的组件之中。

 

Spring中的IoC与DI:

  • IoC是Spring的核心,贯穿始终。对于Spring框架来说,就是由Spring来负责控制对象的生命周期和对象间的关系。

  • Spring中DI有两种注入方式:

    • Setter方式(传值方式)

    • 构造器方式(引用方式)。

10.1 概念

在Spring创建对象的同时,为其属性赋值,称之为依赖注入。

10.2 Set注入

创建对象时,Spring工厂会通过Set方法为对象的属性赋值。

10.2.1 定义目标Bean类型
public class User {
   private Integer id;
   private String password;
   private String sex;
   private Integer age;
   private Date bornDate;
   private String[] hobbys;
   private Set<String> phones;
   private List<String> names;
   private Map<String,String> countries;
   private Properties files;
   //Getters And Setters
}

 

10.2.2 基本类型 + 字符串类型 + 日期类型
<bean id="u1" class="com.qf.spring.part1.injection.User">
   <!--base field-->
   <property name="id" value="1001" />
   <property name="password" value="123456" />
   <property name="sex" value="male" />
   <property name="age" value="20" />
   <property name="bornDate" value="1990/1/1" /><!--注意格式"/"-->
</bean>

 

10.2.3 容器类型
<bean id="u1" class="com.qf.spring.part1.injection.User">   
<!--Array-->
   <property name="hobbys">
       <array>
           <value>Run</value>
           <value>Swim</value>
           <value>Climb</value>
       </array>
   </property>

   <!--Set-->
   <property name="phones">
       <set>
           <value>13777777777</value>
           <value>13888888888</value>
           <value>13999999999</value>
       </set>
   </property>

   <!--List-->
   <property name="names">
       <list>
           <value>tom</value>
           <value>jack</value>
           <value>marry</value>
       </list>
   </property>

   <!--Map-->
   <property name="countries">
       <map>
           <entry key="CN" value="China" />
           <entry key="US" value="America" />
           <entry key="KR" value="Korea" />
       </map>
   </property>
   
   <!--Properties-->
   <property name="files">
       <props>
           <prop key="first">One</prop>
           <prop key="second">Two</prop>
           <prop key="third">Three</prop>
       </props>
   </property>
</bean>

 

10.2.4 自建类型
public class Address {

   private String position;
   private String zipCode;
   
   //....get() And set().....
}

 

<!--次要bean,被作为属性-->
<bean id="addr" class="com.qf.spring.part1.injection.Address">
   <property name="position" value="北京市海淀区" />
   <property name="zipCode" value="100001" />
</bean>

<!--主要bean,操作的主体-->
<bean id="u2" class="com.qf.spring.part1.injection.User">
   <property name="address" ref="addr" /><!--address属性引用addr对象-->
</bean>
<!--次要bean,被作为属性-->
<bean id="userDao" class="com.qf.spring.part1.injection.UserDaoImpl" />

<!--主要bean,操作的主体-->
<bean id="userService" class="com.qf.spring.part1.injection.UserServiceImpl">
   <property name="ud" ref="userDao" /><!--ud属性引用userDao对象-->
</bean>

 

10.3 构造注入【了解】

创建对象时,Spring工厂会通过构造方法为对象的属性赋值。

 

10.3.1 定义目标Bean类型
public class Student {
   
   private Integer id;
   private String name;
   private String sex;
   private Integer age;
 
   //Constructors
public Student(Integer id , String name , String sex , Integer age){
    this.id = id;
  this.name = name;
   this.sex = sex;
   this.age = age;
  }
}

 

10.3.2 注入
 <!--构造注入-->
<bean id="u3" class="com.qf.zcg.spring.day1.t2.ioc.Student">
   <constructor-arg name="id" value="1234" /> <!-- 除标签名称有变化,其他均和Set注入一致 -->
   <constructor-arg name="name" value="tom" />
   <constructor-arg name="age" value="20" />
   <constructor-arg name="sex" value="male" />
</bean>

 

10.4 自动注入【了解】

不用在配置中 指定为哪个属性赋值,及赋什么值.

由spring自动根据某个 "原则" ,在工厂中查找一个bean,为属性注入属性值

public class UserServiceImpl implements UserService {
   private UserDAO userDAO;
   //Getters And Setters
  ....
}
<bean id="userDao" class="com.qf.spring.part1.injection.UserDaoImpl" />
<!-- 为UserServiceImpl中的属性基于类型自动注入值 -->
<bean id="userService" class="com.qf.spring.part1.injection.UserServiceImpl" autowire="byType"></bean>
<bean id="userDao" class="com.qf.spring.part1.injection.UserDaoImpl" />
<!-- 为UserServiceImpl中的属性基于类型自动注入值 -->
<bean id="userService" class="com.qf.spring.part1.injection.UserServiceImpl" autowire="byName"></bean>

 

十一、Bean细节


11.1 控制简单对象的单例、多例(原型)模式

配置< bean scope="singleton | prototype" />

<!--
singleton(默认):每次调用工厂,得到的都是同一个对象。
prototype:每次调用工厂,都会创建新的对象。
-->
<bean id="mc" class="com.qf.zcg.spring.day1.t1.basic.MyClass" scope="singleton" />
  • 注意:需要根据场景决定对象的单例、多例模式。

  • (单例)可以共用:Service、DAO、SqlSessionFactory(或者是所有的工厂)。

  • (多例)不可共用:Connection、SqlSession、ShoppingCart。

 

11.2 FactoryBean创建复杂对象【了解】

作用:让Spring可以创建复杂对象、或者无法直接通过反射创建的对象。

FactoryBean解决复杂对象创建
image-20190419235128663

 

11.2.1 实现FactoryBean接口
接口方法描述
image-20190419234550731
  • 注意:isSingleton方法的返回值,需根据所创建对象的特点决定返回true/false。

  • 例如:Connection 不应该被多个用户共享,返回false。

  • 例如:SqlSessionFactory 重量级资源,不该过多创建,返回true+。

11.2.2 配置spring-context.xml
配置与获取方式
image-20190419235939298

 

11.2.3 特例
获取FactoryBean接口的实现类对象,而非getObject()所生产的对象。
image-20190420000713143

十二、Spring工厂特性


12.1 饿汉式创建优势

工厂创建之后,会将Spring配置文件中的所有对象都创建完成(饿汉式)。

提高程序运行效率。避免多次IO,减少对象创建时间。(概念接近连接池,一次性创建好,使用时直接获取)

12.2 生命周期方法

  • 自定义初始化方法:添加“init-method”属性,Spring则会在创建对象之后,调用此方法。

  • 自定义销毁方法:添加“destroy-method”属性,Spring则会在销毁对象之前,调用此方法。

  • 销毁:工厂的close()方法被调用之后,Spring会毁掉所有已创建的单例对象。

  • 分类:Singleton对象由Spring容器销毁、Prototype对象由JVM销毁。

12.3 生命周期注解

初始化注解、销毁注解

import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;

@PostConstruct //初始化
public void init(){
   System.out.println("init method executed");
}

@PreDestroy //销毁
public void destroy(){
   System.out.println("destroy method executed");
}

12.4 生命周期阶段

单例bean:singleton

随工厂启动创建 ==》 构造方法 ==》 set方法(注入值) ==》 init(初始化) ==》 构建完成 ==》随工厂关闭销毁

多例bean:prototype

被使用时创建 ==》 构造方法 ==》 set方法(注入值) ==》 init(初始化) ==》 构建完成 ==》JVM垃圾回收销毁

 

十三、代理设计模式


13.1 概念

将核心功能与辅助功能(事务、日志、性能监控代码)分离,达到核心业务功能更纯粹、辅助业务功能可复用。

功能分离
image-20190420002535800

13.2 静态代理设计模式

通过代理类的对象,为原始类的对象(目标类的对象)添加辅助功能,更容易更换代理实现类、利于维护。

静态代理
image-20190420004330551
  • 代理类 = 实现原始类相同接口 + 添加辅助功能 + 调用原始类的业务方法。

  • 租房接口

public interface FangDongService {

   public void zufang();
}
  • 租房实现类

// 原始业务类
public class FangDongServiceImpl implements FangDongService{
   @Override
   public void zufang() {
       // 核心功能
       System.out.println("签合同");
       System.out.println("收房租");
  }
}

 

  • 静态代理类 = 实现原始类相同接口 + 添加辅助功能 + 调用原始类的业务方法

// 静态代理类
public class FangDongProxy  implements FangDongService{

   private FangDongService fangDongService = new FangDongServiceImpl();

   @Override
   public void zufang() {

       // 辅助功能、额外功能
       System.out.println("发布租房信息");
       System.out.println("带租客看房");

       //核心= 原始业务类
       fangDongService.zufang();

       // 辅助功能、额外功能
       System.out.println("发布租房信息");
       System.out.println("带租客看房");

  }
}
  • 测试类

public class ProxyTest {

   public static void main(String[] args) {
       FangDongProxy fangDongProxy = new FangDongProxy();
       fangDongProxy.zufang();
  }
}
  • 静态代理的问题

    • 代理类数量过多,不利于项目的管理。

    • 多个代理类的辅助功能代码冗余,修改时,维护性差。

13.3 动态代理设计模式

动态创建代理类的对象,为原始类的对象添加辅助功能。

字节码是在运行期动态生成

 

13.3.1 JDK动态代理实现(基于接口)
public class DynamicProxyTest {
   
   @Test
   public void testJDK(){

       // 目标
       FangDongService fangDongService = new FangDongServiceImpl();

       // 额外功能
       InvocationHandler ih = new InvocationHandler(){
           @Override
           public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
               // 辅助功能、额外功能
               System.out.println("发布租房信息1");
               System.out.println("带租客看房1");
               // 核心
               fangDongService.zufang();
               return null;
          }
      };

       // 动态生成 代理类
       FangDongService proxy = (FangDongService)Proxy.newProxyInstance(DynamicProxyTest.class.getClassLoader(),
                               fangDongService.getClass().getInterfaces(), ih);

       proxy.zufang();
  }
}

 

13.3.2 CGlib动态代理实现(基于继承)
public class DynamicProxyTest {
   @Test
   public void testCGLIB(){
       // 目标
       FangDongService fangDongService = new FangDongServiceImpl();

       //创建字节码增强器,用来给类创建代理
       Enhancer enhancer = new Enhancer();
       //指定被代理类, 将被代理类字节码传入
       enhancer.setSuperclass(FangDongServiceImpl.class);
       //实现代理中的额外功能
       enhancer.setCallback(new org.springframework.cglib.proxy.InvocationHandler() {
           @Override
           public Object invoke(Object o, Method method, Object[] objects) throws Throwable {
               // 辅助功能、额外功能
               System.out.println("发布租房信息2");
               System.out.println("带租客看房2");
               // 核心
               fangDongService.zufang();
               return null;
          }
      });


       // 动态生成代理类
       FangDongServiceImpl proxy = (FangDongServiceImpl)enhancer.create();

       proxy.zufang();
  }
}

 

十四、面向切面编程【重点


14.1 概念

AOP(Aspect Oriented Programming),即面向切面编程,利用一种称为"横切"的技术,剖开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其命名为"Aspect",即切面。所谓"切面",简单说就是那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块之间的耦合度,并有利于未来的可操作性和可维护性。

 

14.2 作用

Spring的AOP编程即是通过动态代理类为原始类的方法添加辅助功能。

 

14.3 AOP开发术语

  • 连接点(Joinpoint):连接点是程序类中客观存在的方法,可被Spring拦截并切入内容。

  • 切入点(Pointcut):被Spring切入连接点。

  • 通知、增强(Advice):可以为切入点添加额外功能,分为:前置通知、后置通知、异常通知、环绕通知等。

  • 目标对象(Target):代理的目标对象(真实对象)

  • 引介(Introduction)[了解]:一种特殊的增强,可在运行期为类动态添加Field和Method。

  • 织入(Weaving):把通知应用到具体的类,进而创建新的代理类的过程。

  • 代理(Proxy):被AOP织入通知后,产生的结果类。

  • 切面(Aspect):由切点和通知组成,将横切逻辑织入切面所指定的连接点中。

 

 

14.4 环境搭建

引入AOP相关依赖

<dependency>
   <groupId>org.springframework</groupId>
   <artifactId>spring-aspects</artifactId>
   <version>5.1.6.RELEASE</version>
</dependency>

spring-context.xml引入AOP命名空间

<?xml version="1.0" encoding="UTF-8"?>
<!--
   schema: 规范
   xxx.xsd
   xml schema definition
-->
<beans xmlns="http://www.springframework.org/schema/beans"
      xmlns:aop="http://www.springframework.org/schema/aop"
      xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
      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>

 

14.5 开发流程

定义原始类

public interface UserService {

   public List<User> queryUsers();
   
   public Integer updateUser(User user);

   public Integer saveUser(User user);

   public Integer deleteUser(Integer id);

}
public class UserServiceImpl implements UserService {

   @Override
   public List<User> queryUsers() {
//       System.out.println("事务控制");
//       System.out.println("日志打印");
       System.out.println("queryUser");
       return new ArrayList<>();
  }
   
   @Override
   public Integer updateUser(User user) {
//       System.out.println("事务控制");
//       System.out.println("日志打印");
       System.out.println("update User");
//       if(1==1) {
//           throw new NullPointerException("test 空指针");
//       }
       return 1;
  }

   @Override
   public Integer saveUser(User user) {
//       System.out.println("事务控制");
//       System.out.println("日志打印");
       System.out.println("Save User");
       return 1;
  }

   @Override
   public Integer deleteUser(Integer id) {
//       System.out.println("事务控制");
//       System.out.println("日志打印");
       System.out.println("delete User");
       return 1;
  }
}

 

定义通知类(添加额外功能)

  • 前置通知类

/**
* 前置通知类
*/
public class MyBeforeAdvice implements MethodBeforeAdvice {

   @Override
   public void before(Method method, Object[] objects, Object o) throws Throwable {
       // 额外功能
       System.out.println("事务控制2");
       System.out.println("日志打印2");
  }
}

 

  • 后置通知,在核心之后执行,如果核心有异常,则不执行

/**
* 后置通知,在核心之后执行,如果核心有异常,则不执行
*/
public class MyAfterAdvice implements AfterReturningAdvice {

   @Override
   public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {
       System.out.println("after!! 后置通知辅助内容!!!");
  }
}

 

  • 环绕通知, 在核心功能前后都执行辅助功能

/**
* 环绕通知
*/
public class MyMethodInterceptor implements MethodInterceptor {

   @Override
   public Object invoke(MethodInvocation invocation) throws Throwable {
       System.out.println("环绕通知: 开始辅助内容tx begin~~");
       Object ret = invocation.proceed();// 触发,执行核心功能
       System.out.println("环绕通知: 结尾复制内容tx end!!");
       return ret;
  }
}

 

  • 异常通知, 在核心中抛异常时,执行

/**
* 在核心中抛异常时,执行
*/
public class MyThrowsAdvice implements ThrowsAdvice {

   public void afterThrowing(Exception ex){
       System.out.println("my throws 异常通知辅助功能");
  }
}

 

定义bean标签

<!-- 目标 : 原始业务-->
<bean id="us" class="com.qf.service.UserServiceImpl" />

<!-- 通知:额外功能 -->
<!-- 前置通知类 -->
<bean id="before" class="com.qf.aop.MyBeforeAdvice"></bean>

<!-- 后置通知,在核心之后执行,如果核心有异常,则不执行 -->
<bean id="after" class="com.qf.aop.MyAfterAdvice"></bean>

<!-- 在核心中抛异常时,执行 -->
<bean id="throws" class="com.qf.aop.MyThrowsAdvice"></bean>

<!-- 环绕通知 -->
<bean id="mi" class="com.qf.aop.MyMethodInterceptor"></bean>

 

定义切入点(PointCut)

形成切面(Aspect)

<aop:config>
   <!-- 切入点【修饰符 返回值 包.类 方法名 参数表 】 -->
   <aop:pointcut id="pc_shine" expression="execution(* queryUsers())"/>
   <aop:pointcut id="pc_shine2" expression="execution(* deleteUser(..))"/>
   <aop:pointcut id="pc_shine3" expression="execution(* updateUser(..))"/>
   <aop:pointcut id="pc_shine4" expression="execution(* saveUser(..))"/>
   
   <!-- 组装 -->
   <aop:advisor advice-ref="before" pointcut-ref="pc_shine"/>
   <aop:advisor advice-ref="after" pointcut-ref="pc_shine2"/>
   <aop:advisor advice-ref="throws" pointcut-ref="pc_shine3"/>
   <aop:advisor advice-ref="mi" pointcut-ref="pc_shine4"/>
</aop:config>

 

测试代码

public class ProcessorTest {

   @Test
   public void testProcessor(){
       ApplicationContext context  = new ClassPathXmlApplicationContext("/spring-context.xml");

       UserService userService = (UserService) context.getBean("us");
       //演示前置通知
       userService.queryUsers();
       System.out.println("===========queryUsers() end======================");

       //演示环绕通知
       userService.saveUser(new User());
       System.out.println("=============saveUser(new User()) end====================");

       //演示异常通知
       userService.updateUser(new User());
       System.out.println("===========updateUser(new User()) end======================");

       //演示后置通知
       userService.deleteUser(1);
       System.out.println("==============deleteUser(1) end===================");
  }
}

 

14.6 AOP小结

  • 通过AOP提供的编码流程,更便利的定制切面,更方便的定制了动态代理。

  • 进而彻底解决了辅助功能冗余的问题;

  • 业务类中职责单一性得到更好保障;

  • 辅助功能也有很好的复用性。

 

14.7 通知类【可选】

定义通知类,达到通知效果0

前置通知:MethodBeforeAdvice

后置通知:AfterReturningAdvice //有异常不执行,方法会因异常而结束,无返回值

异常通知:ThrowsAdvice

环绕通知:MethodInterceptor, 注意:在环绕中必须调用原始方法

 

14.8 通配切入点

根据表达式通配切入点

<aop:config>
   <!--匹配参数-->
   <aop:pointcut id="pc01" expression="execution(* *(com.qf.pojo.User))"/>
   <!--匹配任意方法(无参)-->
   <aop:pointcut id="pc02" expression="execution(* *())"/>
   <!--匹配方法名(任意参数)-->
   <aop:pointcut id="pc03" expression="execution(* saveUser(..))"/>
   <!--匹配返回值类型(任意参数)-->
   <aop:pointcut id="pc04" expression="execution(java.lang.Integer *(..))"/>
   <!--匹配类名(任意参数)-->
   <aop:pointcut id="pc05" expression="execution(* com.qf.service.UserServiceImpl.*(..))"/>
   <!--匹配包名(任意参数)-->
   <aop:pointcut id="pc06" expression="execution(* com.qf.service.*.*(..))"/>
   <!--匹配包名、以及子包名(任意参数)-->
   <aop:pointcut id="pc07" expression="execution(* com..*.*(..))"/>

   <!--组装-->
   <aop:advisor advice-ref="after" pointcut-ref="pc01"/>
</aop:config>

 

14.9 JDK和CGLIB选择

  • spring底层,包含了jdk代理和cglib代理两种动态代理生成机制

  • 基本规则是:目标业务类如果有接口则用JDK代理,没有接口则用CGLib代理

class DefaultAopProxyFactory{
   // 该方法中明确定义了 JDK代理和CGLib代理的选取规则
   // 基本规则是:目标业务类如果有接口则用JDK代理,没有接口则用CGLib代理
   public AopProxy createAopProxy(){...}
}

 

14.10 后处理器

  • spring中定义了很多后处理器;

  • 每个bean在创建完成之前 ,都会有一个后处理过程,即再加工,对bean做出相关改变和调整;

  • spring-AOP中,就有一个专门的后处理器,负责通过原始业务组件(Service),再加工得到一个代理组件。

  • 后处理器是一种特殊的Bean,实现了BeanPostProcessor接口,该接口包含两个方法,bean在初始化的时候都会执行

常用后处理器
系统后处理器
14.10.1 后处理器定义
/**
* 定义bean后处理器
* 作用:在bean的创建之后,进行再加工
*/
public class MyBeanPostProcessor implements BeanPostProcessor{

   /**
    * 在bean的init方法之前执行
    * @param bean 原始的bean对象
    * @param beanName
    * @return
    * @throws BeansException
    */
   public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
       System.out.println("后处理器 在init之前执行~~~"+bean.getClass());
       return bean;
  }
   
/**
    * 在bean的init方法之后执行
    * @param bean postProcessBeforeInitialization返回的bean
    * @param beanName
    * @return
    * @throws BeansException
    */
   @Override
   public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
       System.out.println("后处理器 在init之后执行~~~"+bean.getClass());
       return bean;// 此处的返回是 getBean() 最终的返回值
  }
}
14.10.2 配置后处理器
<!-- 配置后处理器,将对工厂中所有的bean声明周期进行干预 -->
<bean class="com.qf.aop.MyBeanPostProcessor"></bean>
14.10.3 bean生命周期

构造 》 注入属性 满足依赖 》 后处理器前置过程 》 初始化 》后处理器后置过程 》 返回 》 销毁

 

十五、Spring + MyBatis【重点


15.1 配置数据源

将数据源配置到项目中

引入pom.xml中依赖

<dependencies>
   <!-- Spring -->
   <dependency>
       <groupId>org.springframework</groupId>
       <artifactId>spring-context</artifactId>
       <version>5.1.6.RELEASE</version>
   </dependency>
   <dependency>
       <groupId>org.springframework</groupId>
       <artifactId>spring-aspects</artifactId>
       <version>5.1.6.RELEASE</version>
   </dependency>
   <dependency>
       <groupId>org.springframework</groupId>
       <artifactId>spring-test</artifactId>
       <version>5.1.6.RELEASE</version>
   </dependency>
   <dependency>
       <groupId>org.springframework</groupId>
       <artifactId>spring-jdbc</artifactId>
       <version>5.1.6.RELEASE</version>
   </dependency>

   <!-- Spring整合mybatis -->
   <dependency>
       <groupId>org.mybatis</groupId>
       <artifactId>mybatis-spring</artifactId>
       <version>1.3.1</version>
   </dependency>

   <!-- MyBatis -->
   <dependency>
       <groupId>org.mybatis</groupId>
       <artifactId>mybatis</artifactId>
       <version>3.4.5</version>
   </dependency>

   <!-- mysql驱动 依赖 -->
   <dependency>
       <groupId>mysql</groupId>
       <artifactId>mysql-connector-java</artifactId>
       <version>5.1.25</version>
       <scope>runtime</scope>
   </dependency>

   <!-- 连接池 -->
   <dependency>
       <groupId>com.alibaba</groupId>
       <artifactId>druid</artifactId>
       <version>1.1.12</version>
   </dependency>

   <!-- Junit -->
   <dependency>
       <groupId>junit</groupId>
       <artifactId>junit</artifactId>
       <version>4.12</version>
   </dependency>
</dependencies>

 

15.1.1 创建数据库和表
create database mydb1;

create table user(
userId int primary key auto_increment,
username varchar(20),
password varchar(20),
address varchar(20),
phone varchar(20),
birthday date,
money double(10,2)
)
15.1.2 整合Spring配置文件和properties配置文件
  • 引入jdbc.properties配置文件

#jdbc.properties
jdbc.driverClass=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/mydb1?useUnicode=true&characterEncoding=UTF-8
jdbc.username=root
jdbc.password=123456
jdbc.init=3
jdbc.minIdle=3
jdbc.maxActive=10
  • spring-context.xml配置文件

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
      xmlns:aop="http://www.springframework.org/schema/aop"
      xmlns:tx="http://www.springframework.org/schema/tx"
      xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
      xmlns:context="http://www.springframework.org/schema/context"
      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
                          http://www.springframework.org/schema/tx
                          http://www.springframework.org/schema/tx/spring-tx.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">


   <context:property-placeholder location="classpath:jdbc.properties"/>
   <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource" init-method="init" destroy-method="close">
       <!--基本配置-->
       <property name="driverClassName" value="${jdbc.driverClass}"/>
       <property name="url" value="${jdbc.url}"/>
       <property name="username" value="${jdbc.username}"/>
       <property name="password" value="${jdbc.password}"/>

       <!-- 配置初始化大小、最小、最大 -->
       <property name="initialSize" value="${jdbc.init}"/>
       <property name="minIdle" value="${jdbc.minIdle}"/>
       <property name="maxActive" value="${jdbc.maxActive}"/>

       <!-- 配置获取连接等待超时的时间 ms-->
       <property name="maxWait" value="60000"/>

       <!-- 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒 -->
       <property name="timeBetweenEvictionRunsMillis" value="60000"/>

       <!-- 配置一个连接在池中最小生存的时间,单位是毫秒 -->
       <property name="minEvictableIdleTimeMillis" value="300000"/>
   </bean>
   
</beans>

 

15.1.3 dao接口和pojo实体
  • 创建com.qf.pojo实体类包名, 放入下面用户实体对象

public class User {

   private Integer userId;
   private String username;
   private String password;
   private String address;
   private String phone;
   private Date birthday;
   private Integer money;
   
   //......get 和 set方法 ......
}

 

  • 创建com.qf.dao包并放入UserDao接口

public interface UserDao {

   public List<User> findAll();
}

 

  • 在项目resources目录下新建com/qf/dao包, 并新建UserDao.xml文件

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
       "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.qf.dao.UserDao">

   <select id="findAll" resultType="com.qf.pojo.User">
      select * from user
   </select>
</mapper>

 

15.1.4 service接口和实现类
  • 创建com.qf.service包, 新建UserService接口

public interface UserService {

   public List<User> findUserList();
}

 

  • 新建UserServiceImpl实现类

public class UserServiceImpl implements UserService {

   private UserDao userDao;

   public UserDao getUserDao() {
       return userDao;
  }

   public void setUserDao(UserDao userDao) {
       this.userDao = userDao;
  }

   @Override
   public List<User> findUserList() {
       return userDao.findAll();
  }
}

 

15.1.5 Druid监控中心(了解)
<!--web.xml-->
<servlet>
   <servlet-name>DruidStatView</servlet-name>
   <servlet-class>com.alibaba.druid.support.http.StatViewServlet</servlet-class>
</servlet>
<servlet-mapping>
   <servlet-name>DruidStatView</servlet-name>
   <url-pattern>/druid/*</url-pattern>
</servlet-mapping>

 

15.1.6 测试监控中心(了解)

配置tomcat,并访问protocol://ip:port/project/druid/index.html

 

15.2 整合MyBatis

将 SqlSessionFactory、DAO、Service 配置到项目中

15.2.1 配置SqlSessionFactory
<!-- 生产:SqlSessionFactory -->
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
   <!-- 注入连接池 -->
   <property name="dataSource" ref="dataSource"></property>
   <!-- 注入dao-mapper文件信息 ,如果映射文件和dao接口 同包且同名,则此配置可省略-->
   <property name="mapperLocations">
       <list>
           <value>classpath:com/qf/dao/*.xml</value>
       </list>
   </property>
   <!-- 为 dao-mapper文件中的实体 定义缺省包路径
           如:<select id="queryAll" resultType="User"> 中 User类可以不定义包
       -->
   <property name="typeAliasesPackage" value="com.qf.pojo"></property>
</bean>
15.2.2 配置MapperScannerConfigurer

管理DAO实现类的创建,并创建DAO对象,存入工厂管理

  • 扫描所有DAO接口,去构建DAO实现

  • 将DAO实现存入工厂管理

  • DAO实现对象在工厂中的id是:“首字母小写的-接口的类名”,

    例如:UserDAO==>userDAO , OrderDAO==>orderDAO

<!-- mapperScannerConfigurer   userDAO-->
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
   <!-- dao接口所在的包 如果有多个包,可以用逗号或分号分隔
          <property name="basePackage" value="com.a.dao,com.b.dao"></property>
       -->
   <property name="basePackage" value="com.qf.dao"></property>
   <!-- 如果工厂中只有一个SqlSessionFactory的bean,此配置可省略 -->
   <property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"></property>
</bean>
15.2.3 配置Service
<!-- Service -->
<bean id="userService" class="com.qf.service.UserServiceImpl">
   <property name="userDao" ref="userDao"/>
</bean>

 

15.2.4 测试
public class TestUser {

   @Test
   public void testFindUserList() {
       ApplicationContext context  = new ClassPathXmlApplicationContext("/spring-context.xml");
       UserService userService = (UserService) context.getBean("userService");
       List<User> userList = userService.findUserList();
       for (User user : userList) {
           System.out.println(user);
      }
  }
}

 

十六、声明式事务【重点

声明式事务:声明一小段的配置信息,解决事务问题, 在业务方法中不存在任何事务代码

编程式事务:把操作事务的代码写到业务方法中


16.1 事务属性

16.1.1 隔离级别
16.1.1.1 概念

isolation 隔离级别

名称描述
default (默认值)(采用数据库的默认的设置) (建议)
read-uncommited 读未提交
read-commited 读提交 (Oracle数据库默认的隔离级别)
repeatable-read 可重复读 (MySQL数据库默认的隔离级别)
serialized-read 序列化读

隔离级别由低到高为:read-uncommited < read-commited < repeatable-read < serialized-read

16.1.1.2 特性

事务的四个特性: 原子性:要么全部执行,要么全不执行 一致性:事务执行前后逻辑一致 隔离性:事务之间的隔离级别 持久性:一旦提交,永久保存到数据库中

事务隔离级别 读未提交 产生问题:脏读, 不可重复读,幻读(虚读) 不能解决任何问题 级别最低:安全性没有保障,性能最优 读已提交 产生问题:不可重复读,幻读(虚读) 解决问题:脏读 可以重复读 产生问题:幻读(虚读) 解决问题:不可重复读,脏读 串行化 产生问题:无 解决问题:所有问题 级别最高,安全性有保障,效率最低

  • 安全性:级别越高,多事务并发时,越安全。因为共享的数据越来越少,事务间彼此干扰减少。

  • 并发性:级别越高,多事务并发时,并发越差。因为共享的数据越来越少,事务间阻塞情况增多。

16.1.1.3 并发问题

事务并发时的安全问题

问题描述
脏读 一个事务读取到另一个事务还未提交的数据。大于等于 read-commited 可防止
不可重复读 一个事务内多次读取一行数据的相同内容,其结果不一致。大于等于 repeatable-read 可防止
虚读/幻读 一个事务内多次读取一张表中的相同内容,其结果不一致。serialized-read 可防止
16.1.2 传播行为

propagation传播行为

当涉及到事务嵌套(Service调用Service)时,可以设置:

  • SUPPORTS = 不存在外部事务,则不开启新事务;存在外部事务,则合并到外部事务中。(适合查询)

    有没有事务都行,如果有事务则加入事务执行,如果没有事务,非事务运行

    举例:methodA 调用 methodB

    如果methodA有事务, methodB 加入methodA的事务执行

    如果methodA没有事务,methodB则以非事务运行

  • REQUIRED = 不存在外部事务,则开启新事务;存在外部事务,则合并到外部事务中。 (默认值)(适合增删改)

    必须有事务,如果没有事务则创建事务,如果有事务,则加入事务执行

    举例:methodA 调用 methodB

    如果methodA没有事务,methodB 会创建一个新的事务

    如果methodA有事务, methodB会加入methodA的事务

16.1.3 读写性

readonly 读写性

  • true:只读,可提高查询效率。(适合查询)

  • false:可读可写。 (默认值)(适合增删改)

16.1.4 事务超时

timeout事务超时时间

当前事务所需操作的数据被其他事务占用,则等待。

  • 100:自定义等待时间100(秒)。

  • -1:由数据库指定等待时间,默认值。(建议)

16.1.5 事务回滚

rollback-for 回滚属性

  • 如果事务中抛出 RuntimeException,则自动回滚

  • 如果事务中抛出 CheckException(非运行时异常 Exception),不会自动回滚,而是默认提交事务

  • 处理方案 : 将CheckException转换成RuntimException上抛,或 设置 rollback-for="Exception"

 

16.2 配置DataSourceTransactionManager

事务管理器,其中持有DataSource,可以控制事务功能(commit,rollback等)。

<!-- 1. 引入一个事务管理器,其中依赖DataSource,借以获得连接,进而控制事务逻辑 -->
<bean id="tx" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
   <property name="dataSource" ref="dataSource"></property>
</bean>

注意:DataSourceTransactionManager 和 SqlSessionFactoryBean 要注入同一个DataSource的Bean,否则事务控制失败!!!

 

16.3 配置事务通知

基于事务管理器,进一步定制,生成一个额外功能:Advice。

此Advice可以切入任何需要事务的方法,通过事务管理器为方法控制事务。

<tx:advice id="txManager" transaction-manager="tx">
<tx:attributes>
       <!--<tx:method name="insertUser" rollback-for="Exception" isolation="DEFAULT"    
            propagation="REQUIRED" read-only="false"/>-->
       <!-- 以User结尾的方法,切入此方法时,采用对应事务实行-->
       <tx:method name="*User" rollback-for="Exception"/>
       <!-- 以query开头的方法,切入此方法时,采用对应事务实行 -->
       <tx:method name="query*" propagation="SUPPORTS"/>
       <!-- 剩余所有方法 -->
       <tx:method name="*"/>
   </tx:attributes>
</tx:advice>

 

16.4 编织

将事务管理的Advice 切入需要事务的业务方法中

<aop:config>
<aop:pointcut id="pc" expression="execution(* com.qf.service.*.*(..))"/>
   <!-- 组织切面 -->
   <aop:advisor advice-ref="txManager" pointcut-ref="pc"/>
</aop:config>

 

16.5 测试

  • UserDao接口

public interface UserDao {
   public List<User> findAll();
   public void updateUser(User user);
}

 

  • UserDao.xml映射文件

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
       "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.qf.dao.UserDao">

   <select id="findAll" resultType="com.qf.pojo.User">
      select * from user
   </select>

   <update id="updateUser" parameterType="com.qf.pojo.User">
      update user
       <set>
           <if test="username != null">
              username=#{username},
           </if>
           <if test="password != null">
              password=#{password},
           </if>
           <if test="address != null">
              address=#{address},
           </if>
           <if test="phone != null">
              phone=#{phone},
           </if>
           <if test="birthday != null">
              birthday=#{birthday},
           </if>
           <if test="money != null">
              money=#{money},
           </if>
       </set>
      where userId=#{userId}
   </update>
</mapper>

 

  • UserService.java接口

public interface UserService {
   public List<User> findUserList();
   public void updateUser(Integer fromId, Integer toId, Integer money);
}

 

  • UserServiceImpl.java

public class UserServiceImpl implements UserService {

   private UserDao userDao;

   public UserDao getUserDao() {
       return userDao;
  }

   public void setUserDao(UserDao userDao) {
       this.userDao = userDao;
  }

   @Override
   public List<User> findUserList() {
       return userDao.findAll();
  }

   @Override
   public void updateUser(Integer fromId, Integer toId, Integer money) {
       User user1 = new User();
       user1.setUserId(fromId);
       user1.setMoney(money);
       userDao.updateUser(user1);

       //int i = 1/0;

       User user2 = new User();
       user2.setUserId(toId);
       user2.setMoney(money);
       userDao.updateUser(user2);
  }

}

 

  • 测试

@Test
public void testUpdateUser() {
   ApplicationContext context  = new ClassPathXmlApplicationContext("/spring-context.xml");
   UserService userService = (UserService) context.getBean("userService");
   userService.updateUser(1, 2, 500);

}

 

十七、注解开发

17.1 开启注解

spring-context.xml配置文件中设置开启注解使用

<!-- 告知spring注解位置 -->
<context:component-scan base-package="com.qf"></context:component-scan>

 


17.2 声明bean

用于替换自建类型组件的 <bean...>标签;可以更快速的声明bean

  • @Service 业务类专用 @Repository dao实现类专用 @Controller web层专用

  • @Component 通用

  • @Scope 用户控制bean的创建模式

// @Service说明 此类是一个业务类,需要将此类纳入工厂  等价替换掉 <bean class="xxx.UserServiceImpl">
// @Service默认beanId == 首字母小写的类名"userServiceImpl"
// @Service("userService") 自定义beanId为"userService"
@Service //声明bean,且id="userServiceImpl"
@Scope("singleton") //声明创建模式,默认为单例模式 ;@Scope("prototype")即可设置为多例模式
public class UserServiceImpl implements UserService {
...  
}

17.3 注入(DI)

用于完成bean中属性值的注入

  • @Autowired 基于类型自动注入

  • @Resource 基于名称自动注入

  • @Qualifier("userDAO") 限定要自动注入的bean的id,一般和@Autowired联用

  • @Value 注入简单类型数据 (jdk8种+String)

@Service
public class UserServiceImpl implements UserService {
   
   @Autowired //注入类型为UserDAO的bean
   @Qualifier("userDAO2") //如果有多个类型为UserDAO的bean,可以用此注解从中挑选一个
   private UserDAO userDAO;
}
@Service
public class UserServiceImpl implements UserService {
   
@Resource("userDAO3") //注入id=“userDAO3”的bean
   private UserDAO userDAO;
   /*
   @Resource //注入id=“userDAO”的bean
   private UserDAO userDAO;
   */
}
public class XX{
   
   @Value("100") //注入数字
   private Integer id;
   
   @Value("shine") //注入String
private String name;
}

 

17.4 注解所需配置

<!-- 告知spring,哪些包中 有被注解的类、方法、属性 -->
<!-- <context:component-scan base-package="com.qf.a,com.xx.b"></context:component-scan> -->
<context:component-scan base-package="com.qf"></context:component-scan>

<!-- 告知spring,@Transactional在定制事务时,基于tx=DataSourceTransactionManager -->
<tx:annotation-driven transaction-manager="tx"/>

 

17.5 事务控制

用于控制事务切入

  • @Transactional

  • 工厂配置中的 <tx:advice.... 和 <aop:config... 可以省略 !!

//类中的每个方法都切入事务(有自己的事务控制的方法除外)
@Transactional(isolation=Isolation.READ_COMMITTED,propagation=Propagation.REQUIRED,readOnly=false,rollbackFor=Exception.class,timeout = -1)
public class UserServiceImpl implements UserService {
...
   //该方法自己的事务控制,仅对此方法有效
@Transactional(propagation=Propagation.SUPPORTS)
public List<User> queryAll() {
return userDao.queryAll();
}
public void save(User user){
userDao.save(user);
}
}

 

17.6 AOP开发

17.6.1 配置
<!-- 添加如下配置,启用aop注解 -->
<aop:aspectj-autoproxy></aop:aspectj-autoproxy>

 

17.6.2 注解使用
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;

@Aspect // 声明此类是一个切面类:会包含切入点(pointcut)和通知(advice)
@Component //声明组件,进入工厂
public class MyAspect {
   // 定义切入点
   @Pointcut("execution(* com.qf.spring.service.UserServiceImpl.*(..))")
   public void pc(){}
   
   @Before("pc()") // 前置通知
   public void mybefore(JoinPoint a) {
       System.out.println("target:"+a.getTarget());
       System.out.println("args:"+a.getArgs());
       System.out.println("method's name:"+a.getSignature().getName());
       System.out.println("before~~~~");
  }

   @AfterReturning(value="pc()",returning="ret") // 后置通知
   public void myAfterReturning(JoinPoint a,Object ret){
       System.out.println("after~~~~:"+ret);
  }
   
   @Around("pc()") // 环绕通知
   public Object myInterceptor(ProceedingJoinPoint p) throws Throwable {
       System.out.println("interceptor1~~~~");
       Object ret = p.proceed();
       System.out.println("interceptor2~~~~");
       return ret;
  }
   
   @AfterThrowing(value="pc()",throwing="ex") // 异常通知
   public void myThrows(JoinPoint jp,Exception ex){
       System.out.println("throws");
       System.out.println("===="+ex.getMessage());
  }
}

 

 

十八、集成JUnit


18.1 导入依赖

<dependency>
   <groupId>org.springframework</groupId>
   <artifactId>spring-test</artifactId>
   <version>4.3.6.RELEASE</version>
</dependency>
<dependency>
   <groupId>junit</groupId>
   <artifactId>junit</artifactId>
   <version>4.12</version>
</dependency>

18.2 编码

可以免去工厂的创建过程;

可以直接将要测试的组件注入到测试类。

@RunWith(SpringJUnit4ClassRunner.class) //由SpringJUnit4ClassRunner启动测试
@ContextConfiguration("classpath:applicationContext.xml") //spring的配置文件位置
public class SpringTest{//当前测试类也会被纳入工厂中,所以其中属性可以注入

   @Autowired // 注入要测试的组件
   @Qualifier("userDAO")
   private UserDAO userDAO;

   @Test
   public void test(){
       // 测试使用userDAO
       userDAO.queryUser();
      ....
  }
}

 

posted @   ITboy搬砖人  阅读(89)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
· 【译】Visual Studio 中新的强大生产力特性
· 【设计模式】告别冗长if-else语句:使用策略模式优化代码结构
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
点击右上角即可分享
微信分享提示