Spring快速入门

一、概述

Spring是J2EE应用程序框架,是轻量级的IoC(控制反转)和AOP(面向切面)容器框架,主要针对javaBean的生命周期进行管理。使用基本的JavaBean代替了传统的EJB,降低了企业应用开发的复杂性。借助依赖注入、AOP应用、面向接口编程等特性,降低了业务组件之间的耦合度,增强了系统的可扩展性。Spring是非侵入式的,应用中的对象不依赖于Spring的特定类。可以整合和兼容其他框架,让已有的技术和框架更加易用。

官网地址

二、框架结构

Spring框架是一个分层架构,由七个模块组成。Spring的各模块构建在核心容器之上,核心容器定义了创建、配置和管理bean的方式。

  1. Spring Core

    提供了Spring框架的基本功能。核心容器的主要组件是BeanFactory,是工厂模式的实现。BeanFactory使用IoC将应用程序的配置和依赖性规范与实际的应用程序代码分离。

  2. Spring Context

    是一个配置文件,为Spring框架提供上下文信息。Spring上下文包括各种企业服务,如JNDI、EJB、电子邮件、国际化、校验和调度等。

  3. Spring AOP

    通过配置管理特性,该模块将面向方面的编程功能集成到了Spring框架中。为基于Spring的应用程序中的对象提供了事务管理服务,将声明性事务管理集成到应用程序中。

  4. Spring DAO

    提供了有意义的异常层次结构,可用该结构来管理异常处理和错误信息。异常层次结构简化了错误处理,极大地降低了需要编写的异常代码量。

  5. Spring ORM

    Spring框架插入了若干ORM框架,从而提供了ORM的对象关系工具。所有这些框架都遵从Spring的通用事务和DAO异常层次结构。

  6. Spring Web

    Web上下文模块建立在应用程序上下文模块之上,为基于Web的应用程序提供了上下文。Web模块还简化了处理多部分请求以及将请求参数绑定到域对象的工作。

  7. Spring MVC

    SpringMVC框架是一个全功能的构建Web应用程序的MVC实现。

三、核心组件

Spring框架的核心组件主要有三个:Bean组件、Context组件、Core组件。

1、Bean组件

Bean组件在Spring的org.springframework.beans包下,这个包下的所有类主要负责bean的定义、创建、以及对bean的解析。在实际使用中只需要关注bean的创建即可,bean的定义和bean的解析由框架完成。

  • bean的创建使用工厂模式,顶层接口是BeanFactory,最终的实现类是DefaultListableBeanFactory。BeanFactory有三个子接口:ListableBeanFactory、HierarchicalBeanFactory、AutowireCapableBeanFactory。这四个接口共同定义了bean的集合、bean之间的关系、以及bean的行为。
  • bean的定义主要由BeanDefinition描述,描述了配置文件中所定义的<bean>节点中的所有信息。当Spring成功解析节点后,节点信息被转化成BeanDefinition对象,之后所有的操作都是针对该对象。
  • bean的解析主要就是对Spring配置文件的解析。

2、Context组件

Context组件在Spring的org.springframework.context包下,提供一个运行时环境,用以保存各个对象的状态。ApplicationContext是Context的顶级父类,标识了一个应用环境的基本信息。还继承了五个接口,主要扩展Context的功能。

ApplicationContext主要有两个子类:

  • ConfigurableApplicationContext,表示该Context是可修改的,用户在构建Context时可以动态添加或修改已有的配置信息。
  • WebApplicationContext,表示是为web准备的Context,可以直接访问ServletContext。

ApplicationContext必须完成以下几件事:

  • 标识一个应用环境
  • 利用BeanFactory创建Bean对象
  • 保存对象关系表
  • 能够捕获各种事件

3、Core组件

Core组件包含了很多关键类,其中一个重要部分就是定义了资源的访问方式。Resource接口封装了各种可能的资源类型,对使用者屏蔽了文件类型的不同。Resource接口继承了InputStreamSource接口,接口中的getInputStream方法返回InputStream类,所有的资源都可以通过这个类来获取,也屏蔽了资源的提供者。ResourceLoader接口屏蔽了资源的加载者,只需要实现这个接口就可以加载所有类型的资源。Context把资源的加载、解析和描述工作委托给ResourcePatternResolver类来完成,由它整合在一起供其他组件使用。

四、原理

1、IoC

IoC是Spring的核心之一。传统情况下,当一个对象依赖于另一个对象时,需要通过new操作主动创建另一个对象的实例,这样的话,两个对象之间就产生了耦合。通过IoC,一个对象依赖的其他对象会通过被动的方式传递进来,而不是自己创建或查找依赖对象。创建依赖对象实例的工作通常由Spring容器来完成,然后注入给依赖者,所以也被称为DI(依赖注入)。

1.1 IoC容器

BeanFactory是IoC容器的核心接口,负责实例化、定位、配置应用程序中的对象及建立这些对象间的依赖。XMLBeanFactory实现BeanFactory接口,通过获取xml配置文件数据,组成应用对象及对象间的依赖关系。

  • BeanFactoryPostProcessor,在构建BeanFactory时调用
  • BeanPostProcessor,在构建Bean对象时调用
  • InitializingBean,在Bean实例创建时调用
  • DisposableBean,在Bean实例销毁时调用
  • FactoryBean,用来创建Bean实例的Bean

1.2 注入方法

可以通过三种方法进行注入:set方法、构造函数、接口。

set方法注入

可以为属性提供set方法。在配置文件中添加bean标签,属性class的值为对象实现类的完整路径。添加property标签,属性name的值与对象类中对应的属性名称一致,属性value的值就是给对象类中对应的属性注入的值。

实现类:

public class User {
    private String username;

    public String getUsername() {
        return username;
    }
    public void setUsername(String username) {
        this.username= username;
    }
}

配置文件:

<bean id="userAction" class="com.spring.action.User" >  
    <property name="username" value="admin"></property>  
</bean>  

也可以为对象提供set方法。在配置文件中添加对引用对象的bean声明,属性ref的值是所要引用的对象。这样,框架就会将引用对象注入到对象类中。

实现类:

public class UserAction {
    private UserService userservice;

    public void setUserService(UserService userservice) {
        this.userservice = userservice;
    }
}

配置文件:

<bean id="userService" class="com.spring.service.UserService"></bean>
<bean id="userAction" class="com.spring.action.UserAction" >
    <property name="userservice" ref="userService"></property>
</bean>

构造方法注入

直接通过构造函数,将要依赖的属性或对象作为参数传递给构造函数。

实现类:

public class UserAction {
    private String username;

    public UserAction(String username) {
        this.username= username;
    }
}

配置文件:

<bean id="userAction" class="com.spring.action.UserAction" >
    <constructor-arg constructor-argvalue="admin"></constructor-arg>
</bean>

接口方法注入

并不常用,所以不做介绍。

1.3 自动装配

修改Spring配置文件中bean标签的autowire属性,可以进行自动装配,简化配置,提高开发效率。自动装配属性有六个值,分别代表不同的含义:

  • byName

    从Spring环境中获取目标对象时,目标对象中的属性会根据名称在整个Spring环境中查找bean标签的id属性值。如果有相同的,则获取这个对象,实现关联。要保证id不能重复。

  • byType

    从Spring环境中获取目标对象时,目标对象中的属性会根据名称在整个Spring环境中查找bean标签的class属性值。如果有相同的,则获取这个对象,实现关联。当存在多个相同类型的bean对象时,会出错。

  • constructor

    使用构造方法完成对象注入,也就是根据构造方法的参数类型进行对象查找,相对于采用byType方式。

  • autodetect

    如果对象没有无参数的构造方法,则自动选择constructor的装配方式进行构造注入;如果对象含有无参数的构造方法,则自动选择byType的装配方式进行set注入。

  • default

    默认采用上一级标签的自动装配的值。

  • no

    不进行自动装配。

2、AOP

AOP是Spring的另一个核心。Spring通过AOP,允许分离应用的业务逻辑与系统级服务(如日志、事务、权限等),从而提高了内聚性。

2.1 AOP与OOP

AOP即面向切面编程,与OOP面向对象编程相辅相成。OOP要求职责分离,不同的类完成不同的功能,降低了代码的复杂度,便于重用。但在职责分离的同时,如果多个类需要进行相同操作时,需要在每个类中都加入这些内容,增加了代码的重复性。如果将这些内容单独封装在一个类中,再通过每个类进行调用,会增加类之间的耦合度。AOP就是在程序运行时,动态地将这些内容切入到类的指定方法、指定位置上。OOP中,基本单元是类;AOP中,基本单元是切面。OOP横向分成各个类,AOP纵向给对象加入特定代码。

2.2 AOP术语

AOP中包含很多术语,理解这些术语才能全面的理解AOP的真正含义。

  • 切面(Aspect),横切面的功能,用于抽象出类或接口

    由Advice和Pointcut组成,既包含了横切面的定义,也包含了切入点的定义。可以使用@Aspect注解实现。

  • 增强(Advice),横切面功能的具体实现,对象的实例

    由Aspect添加到特定Joinpoint的代码,

  • 连接点(Joinpoint),程序运行期间的时间节点

    在Spring AOP中,Joinpoint总是表现为方法执行。

  • 切入点(Pointcut),描述横切面功能应用的限制

    使用Pointcut的目的就是提供一组规则来匹配Joinpoint,只为满足条件的Joinpoint添加Advice代码。

  • 代理(Proxy),织入Advice的结果类

    Spring AOP默认使用JDK动态代理为接口实现代理。如果业务逻辑对象没有实现接口,则使用CGLIB代理。Spring AOP建议基于接口编程,对接口进行AOP,而不是基于类。

  • 目标(Target),业务操作的实际对象,织入Advice的目标对象,也被称为被增强的对象

    因为Spring AOP是通过运行时代理的方式实现Aspect的,所以目标对象总是一个代理对象。一个类被AOP织入Advice后,会产生一个结果类,是包含原类和增强逻辑的代理类。

  • 织入(Weave),将组件应用到业务流程中的过程,将Aspect和其他对象关联并创建目标对象的过程

    Spring采用动态代理织入,而AspectJ则采用编译器和类装载器织入。

2.3 用法

定义Aspect

当使用注解@Aspect标注一个Bean后,Spring框架会自动将其添加到Spring AOP中。要想将对象转换成Bean,还需要使用@Component之类的注解。如果一个类被@Aspect标注,那么这个类就不能成为其他Aspect的目标对象了。

@Component
@Aspect
public class Test{
}

声明Pointcut

Pointcut声明由两部分组成:一个pointcut表达式,一个方法签名。使用注解@Pointcut标注pointcut表达式。

@Pointcut("execution(* com.service.UserService.*(..))")
private void dataAccessOperation() {}

该声明描述的是:匹配com.service.UserService下的所有方法的执行。

pointcut表达式由指示器和操作参数组成。指示器有以下几种:

  • execution,匹配joinpoint的执行
  • within,匹配特定类下的所有joinpoint的执行
  • this,匹配一个bean,该bean是一个给定类型的实例
  • target,匹配一个目标对象,该对象是一个给定类型的实例
  • bean,匹配指定bean下的所有方法
  • args,匹配参数满足要求的方法
  • @annotation,匹配由指定注解所标注的方法

声明Advice

Advice通常是和Pointcut表达式关联在一起使用的。

@Component
@Aspect
public class BeforeAspectTest {
    // 定义一个Pointcut, 使用切点表达式函数来描述对哪些Joinpoint使用advise
    @Pointcut("execution(* com.service.UserService.*(..))")
    public void dataAccessOperation() {
    }
}

@Component
@Aspect
public class AdviseDefine {
    // 定义advise
    @Before("com.aspect.PointcutDefine.dataAccessOperation()")
    public void doBeforeAccessCheck(JoinPoint joinPoint) {
        System.out.println("Before advise, method: " + joinPoint.getSignature().toShortString());
    }
}

3、注解

注解Annotation是一种类似注释的机制,在代码中添加注解可以在之后的某个时间使用这些信息。Java虚拟机不会对注解进行编译,而是通过反射机制读取注解中的信息。使用注解的主要目的是为了提高开发效率,不需要配置过多的xml文件,不必在配置文件中添加标签。

3.1 使用注解

需要在配置文件中增加命名空间和约束文件:

<beans ...  
xmlns:context="http://www.springframework.org/schema/context"  
xsi:schemaLocation="  
...  
http://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context-2.5.xsd  
">  

3.2 配置基包

告诉框架哪些类使用注解:

<context:component-scan base-package="com.spring" />

3.3 注解种类

Spring框架自带很多注解。

持久层注解@Repository

@Repository
public class UserDao{
    // ......
}

等同于配置文件中的:

<bean id="userDao" class="com.spring.UserDao" />

服务层注解@Service

@Service(value="testService")
public classTestService {

    @Resource //相当于自动装配
    private UserDao userDao;
   
    public UserDao getUserDao() {
        return userDao;
    }

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

等同于配置文件中的:

<bean id="testService" class="com.spring.UserService" />

其中,@Resource表示对象间的关系,默认采用byName方式进行装配,如果找不到关联对象,则采用byType方式装配。

控制层注解@Controller

@Controller(value="ua")
@Scope(value="prototype")
public class UserAction {

    @Resource
    private UserService userService;

    public UserService getUserService() {
        return userService;
    }
}

等同于配置文件中的:

<bean id="ua" class="com.spring.UserAction " />

Tips:以上三层中的组件关键字都可以使用@Component来代替。

从Spring环境中获取Action对象

ServletContext application =request.getSession().getServletContext();
ApplicationContext ac = WebApplicationContextUtils.getWebApplicationContext(application);

// 获取控制层对象
UserAction useraction = (UserAction)ac.getBean("ua");

// 设置编码
response.setContentType("text/html;charset=GBK");
PrintWriter out = response.getWriter();

// 分别打印三个层的对象
out.println("Action:" + userAction);
out.println("Service:" + userAction.getUserService());
out.println("Dao:" + userAction.getUserService().getUserDao());
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,033评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,725评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,473评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,846评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,848评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,691评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,053评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,700评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,856评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,676评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,787评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,430评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,034评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,990评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,218评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,174评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,526评论 2 343

推荐阅读更多精彩内容