Spring:入门

Spring介绍

Spring诞生:

  • 创建Spring的目的就是用来替代更加重量级的的企业级Java技术
  • 简化Java的开发
    • 基于POJO轻量级和最小侵入式开发
    • 通过依赖注入和面向接口实现松耦合
    • 基于切面和惯例进行声明式编程
    • 通过切面和模板*减少样板式代码 *

侵入式概念

侵入式框架

  • 对于EJB、Struts2等一些传统的框架,

    通常是要实现特定的接口,继承特定的类才能增强功能

    • 改变了java类的结构

非侵入式

  • 对于Hibernate、Spring等框架,对现有的类结构没有影响,就能够增强JavaBean的功能

松耦合

前面我们在写程序的时候,都是面向接口编程,通过DaoFactroy等方法来实现松耦合

这里写图片描述

DAO层和Service层通过DaoFactory来实现松耦合

  • 如果Serivce层直接new DaoBook(),那么DAO和Service就紧耦合了【Service层依赖紧紧依赖于Dao】

而Spring给我们更加合适的方法来实现松耦合,并且更加灵活、功能更加强大!—->IOC控制反转

切面编程

切面编程也就是AOP编程,其实我们在之前也接触过…动态代理就是一种切面编程了

当时我们使用动态代理+注解的方式给Service层的方法添加权限.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
@Override
@permission("添加分类")
/*添加分类*/
public void addCategory(Category category) {
categoryDao.addCategory(category);
}


/*查找分类*/
@Override
public void findCategory(String id) {
categoryDao.findCategory(id);
}

@Override
@permission("查找分类")
/*查看分类*/
public List<Category> getAllCategory() {
return categoryDao.getAllCategory();
}

/*添加图书*/
@Override
public void addBook(Book book) {
bookDao.addBook(book);

}
  • Controller调用Service的时候,Service返回的是一个代理对象
  • 代理对象得到Controller想要调用的方法,通过反射来看看该方法上有没有注解
  • 如果有注解的话,那么就判断该用户是否有权限来调用 此方法,如果没有权限,就抛出异常给Controller,Controller接收到异常,就可以提示用户没有权限了。

AOP编程可以简单理解成:在执行某些代码前,执行另外的代码

  • Struts2的拦截器也是面向切面编程【在执行Action业务方法之前执行拦截器】

Spring也为我们提供更好地方式来实现面向切面编程

Spring概述

Spring框架目标是简化Java企业级应用开发,并通过POJO为基础的编程模型促进良好的编程习惯。

Spring是分层的Java SE/EE应用一站式的轻量级开源框架,以IOC和AOP为内核,提供了展现层Spring MVC、持久层Spring JDBC以及业务层事务管理等一站式的企业级应用技术。

spring优点

  • 轻量:Spring 是轻量的,基本的版本大约2MB。
  • 方便解耦,简化开发。通过IOC容器,用户可以将对象间的依赖关系交由Spring进行控制,避免硬编码造成的过度耦合。
    • Ioc使得用户不必再为单例模式类、属性文件解析等底层需求编写代码,可以更专注于上层应用。
  • 面向切面的编程(AOP):Spring支持面向切面的编程,并且把应用业务逻辑和系统服务分开。
  • 容器:Spring 包含并管理应用中对象的生命周期和配置。
  • MVC框架:Spring的WEB框架是个精心设计的框架,是Web框架的一个很好的替代品。
  • 声明式事务的支持:Spring支持用户以声明的方式进行事务,Spring支持事务管理,提供一个持续的事务管理接口,可以扩展到上至本地事务下至全局事务(JTA)。
  • 异常处理:Spring 提供方便的API把具体技术相关的异常(比如由JDBC,Hibernate or JDO抛出的)转化为一致的unchecked 异常。

Spring模块组成

这里写图片描述

Spring可以分为6大模块:

  • Spring Core:Spring的核心功能: IOC容器, 解决对象创建及依赖关系。

  • Spring Web:Spring对web模块的支持。

    • 1
      可以与struts整合,让struts的action创建交给spring
    • 1
      spring mvc模式
  • Spring DAO:Spring 对jdbc操作的支持 【JdbcTemplate模板工具类】

  • Spring ORM:spring对orm的支持:

    • 既可以与hibernate整合,【session】。
    • 也可以使用spring的对hibernate操作的封装。
  • Spring AOP:切面编程。

  • SpringEE:spring 对javaEE其他模块的支持。

并且

  • 核心容器
    • Core module
    • Bean module
    • Context module
    • Expression Language module
  • 数据集成/访问。提供与数据库交互的支持
    • JDBC module
    • ORM module
    • OXM module
    • Java Messaging Service(JMS) module
    • Transaction module
  • Web应用程序的支持
    • Web module
    • Web-Servlet module
    • Web-Struts module
    • Web-Portlet module
  • AOP
  • Instrumentation:类检测和类加载器支持
  • Test
  • 杂项
    • Messaging
    • Aspects

什么是 Spring 配置文件?

Spring 配置文件是 XML 文件。该文件主要包含类信息。它描述了这些类是如何配置以及相互引入的。但是,XML 配置文件冗长且更加干净。如果没有正确规划和编写,那么在大项目中管理变得非常困难。

Spring 应用程序有哪些不同组件?

Spring 应用一般有以下组件:

  • 接口 - 定义功能。
  • Bean 类 - 它包含属性,setter 和 getter 方法,函数等。
  • Spring 面向切面编程(AOP) - 提供面向切面编程的功能。
  • Bean 配置文件 - 包含类的信息以及如何配置它们。
  • 用户程序 - 它使用接口。

使用 Spring 有哪些方式?

使用 Spring 有以下方式:

  • 作为一个成熟的 Spring Web 应用程序。
  • 作为第三方 Web 框架,使用 Spring Frameworks 中间层。
  • 用于远程使用。
  • 作为企业级 Java Bean,它可以包装现有的 POJO(Plain Old Java Objects)。

模块概述

IOC

Spring核心模块实现了IOC的功能,它将类与类之间的依赖从代码中脱离出来,用配置的方式进行依赖关系描述,由IOC容器负责依赖类间的创建、拼接、管理、获取等工作。

BeanFactory接口是Spring框架的核心接口,实现了容器的许多核心功能。

Context模块构建于核心模块上,扩展了BeanFactory的功能,添加了il8n国际化、Bean生命周期控制、框架事件体系、资源架子啊透明化等多项功能。并提供许多企业级服务的支持,如邮件服务、任务调度、JNDI获取、EJB集成、远程访问等,ApplicationContext是Context模块的核心接口。

引出Spring

我们试着回顾一下没学Spring的时候,是怎么开发Web项目的

    1. 实体类—>class User{ }
    1. daoclass–> UserDao{ .. 访问**db}
    1. service—>class UserService{ UserDao userDao = new UserDao();}
    1. actionclass UserAction{UserService userService = new UserService();}

用户访问:

  • Tomcat->action->service->dao

我们来思考几个问题:

  • ①:对象创建创建能否写死?

  • ②:对象创建细节

    • 对象数量

      • 1
        action  多个   【维护成员变量】
      • 1
        service 一个   【不需要维护公共变量】
      • 1
        dao     一个   【不需要维护公共变量】
    • 创建时间

      • 1
        action    访问时候创建
      • 1
        service   启动时候创建
      • 1
        dao       启动时候创建
  • ③:对象的依赖关系

    • action 依赖 service
    • service依赖 dao

对于第一个问题和第三个问题,我们可以通过DaoFactory解决掉(虽然不是比较好的解决方法)

对于第二个问题,我们要控制对象的数量和创建事件就有点麻烦了….

Spring框架通过IOC就很好地可以解决上面的问题….

IOC控制反转

Spring的核心思想之一:Inversion of Control , 控制反转 IOC

那么控制反转是什么意思呢???对象的创建交给外部容器完成,这个就做控制反转。

  • Spring使用控制反转来实现对象不用在程序中写死
  • 控制反转解决对象处理问题【把对象交给别人创建】

那么对象的对象之间的依赖关系Spring是怎么做的呢??依赖注入,dependency injection.即DI

  • Spring使用依赖注入来实现对象之间的依赖关系
  • 在创建完对象之后,对象的关系处理就是依赖注入

上面已经说了,控制反转是通过外部容器完成的,而Spring又为我们提供了这么一个容器,我们一般将这个容器叫做:IOC容器.

无论是创建对象、处理对象之间的依赖关系、对象创建的时间还是对象的数量,我们都是在Spring为我们提供的IOC容器上配置对象的信息就好了。

那么使用IOC控制反转这一思想有什么作用呢???我们来看看一些优秀的回答…

来自知乎:www.zhihu.com/question/23…

我摘取一下核心的部分:

ioc的思想最核心的地方在于,资源不由使用资源的双方管理,而由不使用资源的第三方管理,这可以带来很多好处。第一,资源集中管理,实现资源的可配置和易管理第二,降低了使用资源双方的依赖程度,也就是我们说的耦合度

也就是说,甲方要达成某种目的不需要直接依赖乙方,它只需要达到的目的告诉第三方机构就可以了,比如甲方需要一双袜子,而乙方它卖一双袜子,它要把袜子卖出去,并不需要自己去直接找到一个卖家来完成袜子的卖出。它也只需要找第三方,告诉别人我要卖一双袜子。这下好了,甲乙双方进行交易活动,都不需要自己直接去找卖家,相当于程序内部开放接口,卖家由第三方作为参数传入。甲乙互相不依赖,而且只有在进行交易活动的时候,甲才和乙产生联系。反之亦然。这样做什么好处么呢,甲乙可以在对方不真实存在的情况下独立存在,而且保证不交易时候无联系,想交易的时候可以很容易的产生联系。甲乙交易活动不需要双方见面,避免了双方的互不信任造成交易失败的问题。因为交易由第三方来负责联系,而且甲乙都认为第三方可靠。那么交易就能很可靠很灵活的产生和进行了。这就是ioc的核心思想。生活中这种例子比比皆是,支付宝在整个淘宝体系里就是庞大的ioc容器,交易双方之外的第三方,提供可靠性可依赖可灵活变更交易方的资源管理中心。另外人事代理也是,雇佣机构和个人之外的第三方。 ==========================update===========================

在以上的描述中,诞生了两个专业词汇,依赖注入和控制反转所谓的依赖注入,则是,甲方开放接口,在它需要的时候,能够将乙方传递进来(注入)所谓的控制反转,甲乙双方不相互依赖,交易活动的进行不依赖于甲乙任何一方,整个活动的进行由第三方负责管理。

参考优秀的博文①:www.tianmaying.com/tutorial/sp…

参考优秀的博文②:这里写链接内容

知乎@Intopass的回答:

  1. 不用自己组装,拿来就用。
  2. 享受单例的好处,效率高,不浪费空间。
  3. 便于单元测试,方便切换mock组件。
  4. 便于进行AOP操作,对于使用者是透明的。
  5. 统一配置,便于修改。

Spring事务

事务的七种传播属性

所谓事务传播行为就是多个事务方法相互调用时,事务如何在这些方法间传播。Spring支持以下7种事务传播行为。

传播行为

含义

PROPAGATION_REQUIRED(XML文件中为REQUIRED)

表示当前方法必须在一个具有事务的上下文中运行,如有客户端有事务在进行,那么被调用端将在该事务中运行,否则的话重新开启一个事务。(如果被调用端发生异常,那么调用端和被调用端事务都将回滚)

PROPAGATION_SUPPORTS(XML文件中为SUPPORTS)

表示当前方法不必需要具有一个事务上下文,但是如果有一个事务的话,它也可以在这个事务中运行

PROPAGATION_MANDATORY(XML文件中为MANDATORY)

表示当前方法必须在一个事务中运行,如果没有事务,将抛出异常

PROPAGATION_NESTED(XML文件中为NESTED)

表示如果当前方法正有一个事务在运行中,则该方法应该运行在一个嵌套事务中,被嵌套的事务可以独立于被封装的事务中进行提交或者回滚。如果封装事务存在,并且外层事务抛出异常回滚,那么内层事务必须回滚,反之,内层事务并不影响外层事务。如果封装事务不存在,则同PROPAGATION_REQUIRED的一样

PROPAGATION_NEVER(XML文件中为NEVER)

表示当方法务不应该在一个事务中运行,如果存在一个事务,则抛出异常

PROPAGATION_REQUIRES_NEW(XML文件中为REQUIRES_NEW)

表示当前方法必须运行在它自己的事务中。一个新的事务将启动,而且如果有一个现有的事务在运行的话,则这个方法将在运行期被挂起,直到新的事务提交或者回滚才恢复执行。

PROPAGATION_NOT_SUPPORTED(XML文件中为NOT_SUPPORTED)

表示该方法不应该在一个事务中运行。如果有一个事务正在运行,他将在运行期被挂起,直到这个事务提交或者回滚才恢复执行

PROPAGATION_NESTED 与PROPAGATION_REQUIRES_NEW的区别:
它们非常类似,都像一个嵌套事务,如果不存在一个活动的事务,都会开启一个新的事务。
使用 PROPAGATION_REQUIRES_NEW时,内层事务与外层事务就像两个独立的事务一样,一旦内层事务进行了提交后,外层事务不能对其进行回滚。两个事务互不影响。两个事务不是一个真正的嵌套事务。同时它需要JTA事务管理器的支持。
使用PROPAGATION_NESTED时,外层事务的回滚可以引起内层事务的回滚。而内层事务的异常并不会导致外层事务的回滚,它是一个真正的嵌套事务。DataSourceTransactionManager使用savepoint支持PROPAGATION_NESTED时,需要JDBC 3.0以上驱动及1.4以上的JDK版本支持。其它的JTATrasactionManager实现可能有不同的支持方式。
PROPAGATION_REQUIRES_NEW 启动一个新的, 不依赖于环境的 “内部” 事务. 这个事务将被完全 commited 或 rolled back 而不依赖于外部事务, 它拥有自己的隔离范围, 自己的锁, 等等. 当内部事务开始执行时, 外部事务将被挂起, 内务事务结束时, 外部事务将继续执行。
另一方面, PROPAGATION_NESTED 开始一个 “嵌套的” 事务, 它是已经存在事务的一个真正的子事务. 潜套事务开始执行时, 它将取得一个 savepoint. 如果这个嵌套事务失败, 我们将回滚到此 savepoint. 潜套事务是外部事务的一部分, 只有外部事务结束后它才会被提交。
由此可见, PROPAGATION_REQUIRES_NEW 和 PROPAGATION_NESTED 的最大区别在于, PROPAGATION_REQUIRES_NEW 完全是一个新的事务, 而 PROPAGATION_NESTED 则是外部事务的子事务, 如果外部事务 commit, 嵌套事务也会被 commit, 这个规则同样适用于 roll back.
REQUIRED,REQUIRES_NEW,NESTED异同
PROPAGATION_NESTED,这是一个嵌套事务,使用JDBC 3.0驱动时,仅仅支持DataSourceTransactionManager作为事务管理器。
需要JDBC 驱动的java.sql.Savepoint类。使用PROPAGATION_NESTED,还需要把PlatformTransactionManager的nestedTransactionAllowed属性设为true(属性值默认为false)。

嵌套事务一个非常重要的概念就是内层事务依赖于外层事务。外层事务失败时,会回滚内层事务所做的动作。而内层事务操作失败并不会引起外层事务的回滚。

NESTED和REQUIRED修饰的内部方法都属于外围方法事务,如果外围方法抛出异常,这两种方法的事务都会被回滚。但是REQUIRED是加入外围方法事务,所以和外围事务同属于一个事务,一旦REQUIRED事务抛出异常被回滚,外围方法事务也将被回滚。而NESTED是外围方法的子事务,有单独的保存点,所以NESTED方法抛出异常被回滚,不会影响到外围方法的事务。
NESTED和REQUIRES_NEW都可以做到内部方法事务回滚而不影响外围方法事务。但是因为NESTED是嵌套事务,所以外围方法回滚之后,作为外围方法事务的子事务也会被回滚。而REQUIRES_NEW是通过开启新的事务实现的,内部事务和外围事务是两个事务,外围事务回滚不会影响内部事务。

参考

  1. Spring:7种事务传播行为