Spring的简单应用与基本原理

 一:重要概念理解

Spring很简单,一定不要想得太复杂,只是有些东西很拗口而已

1:IOC(控制反转)

概念:利用反射的原理将对象创建的权利交给了Spring,Spring在运行的时候根据配置文件(或则注解)来动态的创建和维护对象之间的关系,实现了松散耦合的思想。即创建对象的控制权(即Person person = new Person();方式)由对象本身转向容器(依据配置文件或者注解动态创建)

应用:项目中的Bean都可以交给Spring容器来维护,这样Bean的创建以及销毁以及生命周期都由Spring来处理

2:AOP(面向切面编程)

 概念:通过配置可以实现把业务逻辑和系统服务分离,业务逻辑只关心业务处理而不用去处理其他事情

应用:事务,日志,权限等

3:DI(依赖注入)

概念:由Spring容器将对象注入到使用它的地方,被注入的对象只提供对应的方法接受就行,由容器决定对象之间的依赖关系。

应用:比如Service层需要调用Dao层访问数据库,这时就可以把Dao层的Bean交给Spring进行管理,我们只需要在Service中定义对应的方法来接受由Spring容器负责注入的Dao层的Bean即可。

 

二:Spring的包详解

 

1、Core Container - 核心容器

spring-core:Spring中的核心工具类包。

spring-beans:Spring中定义bean的组件。

spring-context:Spring的运行容器。

spring-context-support:Spring容器的扩展支持。

spring-expression:Spring的表达式语言支持。

2、AOP - 面向切面编程

 

spring-aop:基于代理的AOP支持。

 

spring-aspects:集成Aspects的AOP支持。

 

3、WEB(MVC)

 

spring-web:提供web的基础功能。

 

spring-webmvc:提供springmvc的功能。

 

spring-websocket:提供web socket支持。

 

spring-webmvc-portlet:提供Portlet环境的支持。

 

4、Data Access/Integration - 数据访问/集成

 

spring-jdbc:提供对jdbc连接的封装功能。

 

spring-tx:提供对事务的支持。

 

spring-orm:提供对象-关系映射支持。

 

spring-oxm:提供对象-XML映射支持。

 

spring-jms:提供消息队列的支持。

 

5、Test - 测试

 

spring-test:提供对测试功能的支持。

 

三:常见使用

1:Spring的配置文件中的内容

开启事务注解驱动

事务管理器

开启注解功能,并配置扫描包

配置数据库

配置SQL会话工厂,别名,映射文件

不用编写Dao层的实现类

2:Spring配置声明式事务控制

  • 基于tx和aop名字空间的xml配置文件

配置文件方式:

配置事务管理器

事务的策略(这里可以配置事务的隔离级别,传播属性,是否可读等)

配置事务的切入点,注入事务属性

  • 基于@Transactional注解

注解方式:

配置事务管理器

开启事务控制的注解支持

在类或则方法上面添加@Transaction

事务的属性都在改注解的属性上设置

 

 四:必须要掌握的Spring 常用注解

1.声明bean的注解

@Component 组件,没有明确的角色

@Service 在业务逻辑层使用(service层)

@Repository 在数据访问层使用(dao层)

@Controller 在展现层使用,控制器的声明(C)

2.注入bean的注解

@Autowired:由Spring提供,可以注解在set方法和属性上,推荐注解在属性上

3.java配置类相关注解

@Configuration 声明当前类为配置类,相当于xml形式的Spring配置(注解在类上)

@Bean 注解在方法上,声明当前方法的返回值为一个bean,替代xml中的方式(注解在方法上)

@Configuration 声明当前类为配置类,其中内部组合了@Component注解,表明这个类是一个bean(类上)

@ComponentScan 用于对Component进行扫描,相当于xml中的(类上)

@WishlyConfiguration 为@Configuration与@ComponentScan的组合注解,可以替代这两个注解

4.切面(AOP)相关注解

Spring支持AspectJ的注解式切面编程。

@Aspect 声明一个切面(类上)

使用@After、@Before、@Around定义建言(advice),可直接将拦截规则(切点)作为参数。

@After 在方法执行之后执行(方法上) 
@Before 在方法执行之前执行(方法上) 
@Around 在方法执行之前与之后执行(方法上)

@PointCut 声明切点 
在配置类中使用@EnableAspectJAutoProxy注解开启Spring对AspectJ代理的支持(类上)

5.@Bean的属性支持

@Scope 设置 Spring 容器如何新建Bean实例(方法上,得有@Bean)其设置类型包括:

Singleton (单例,一个Spring容器中只有一个bean实例,默认模式), 
Protetype (每次调用新建一个bean), 
Request (web项目中,给每个http request新建一个bean), 
Session (web项目中,给每个http session新建一个bean), 
GlobalSession(给每一个 global http session新建一个Bean实例)

@StepScope 在Spring Batch中还有涉及

@PostConstruct 由JSR-250提供,在构造函数执行完之后执行,等价于xml配置文件中bean的initMethod

@PreDestory 由JSR-250提供,在Bean销毁之前执行,等价于xml配置文件中bean的destroyMethod

6.@Value注解

@Value 为属性注入值(属性上) 可以是普通字符,表达式,文件资源,配置文件

7.环境切换

@Profile 通过设定Environment的ActiveProfiles来设定当前context需要使用的配置环境。(类或方法上)

@Conditional Spring4中可以使用此注解定义条件话的bean,通过实现Condition接口,并重写matches方法,从而决定该bean是否被实例化。(方法上)

8.异步相关

@EnableAsync 配置类中,通过此注解开启对异步任务的支持,叙事性AsyncConfigurer接口(类上),点击这里了解使用详情。

@Async 在实际执行的bean方法使用该注解来申明其是一个异步任务(方法上或类上所有的方法都将异步,需要@EnableAsync开启异步任务)

9.定时任务相关

@EnableScheduling 在配置类上使用,开启计划任务的支持(类上)

@Scheduled 来申明这是一个任务,包括cron,fixDelay,fixRate等类型(方法上,需先开启计划任务的支持)

10.@Enable*注解说明

这些注解主要用来开启对xxx的支持。 
@EnableAspectJAutoProxy 开启对AspectJ自动代理的支持

@EnableAsync 开启异步方法的支持

@EnableScheduling 开启计划任务的支持

@EnableWebMvc 开启Web MVC的配置支持

@EnableConfigurationProperties 开启对@ConfigurationProperties注解配置Bean的支持

@EnableJpaRepositories 开启对SpringData JPA Repository的支持

@EnableTransactionManagement 开启注解式事务的支持

@EnableTransactionManagement 开启注解式事务的支持

@EnableCaching 开启注解式的缓存支持

11.测试相关注解

@RunWith 运行器,Spring中通常用于对JUnit的支持

@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration 用来加载配置ApplicationContext,其中classes属性用来加载配置类

@ContextConfiguration(classes={TestConfig.class})

12.SpringMVC相关注解

@EnableWebMvc 在配置类中开启Web MVC的配置支持,如一些ViewResolver或者MessageConverter等,若无此句,重写WebMvcConfigurerAdapter方法(用于对SpringMVC的配置)。

@Controller 声明该类为SpringMVC中的Controller

@RequestMapping 用于映射Web请求,包括访问路径和参数(类或方法上)

@ResponseBody 支持将返回值放在response内,而不是一个页面,通常用户返回json数据(返回值旁或方法上)

@RequestBody 允许request的参数在request体中,而不是在直接连接在地址后面。(放在参数前)

@PathVariable 用于接收路径参数,比如@RequestMapping(“/hello/{name}”)申明的路径,将注解放在参数中前,即可获取该值,通常作为Restful的接口实现方法。

@RestController 该注解为一个组合注解,相当于@Controller和@ResponseBody的组合,注解在类上,意味着,该Controller的所有方法都默认加上了@ResponseBody。

@ControllerAdvice 通过该注解,我们可以将对于控制器的全局配置放置在同一个位置,注解了@Controller的类的方法可使用@ExceptionHandler、@InitBinder、@ModelAttribute注解到方法上, 
这对所有注解了 @RequestMapping的控制器内的方法有效。

@ExceptionHandler 用于全局处理控制器里的异常

@InitBinder 用来设置WebDataBinder,WebDataBinder用来自动绑定前台请求参数到Model中。

@ModelAttribute 本来的作用是绑定键值对到Model里,在@ControllerAdvice中是让全局的@RequestMapping都能获得在此处设置的键值对。

 

四:Spring AOP的底层原理

Spring AOP的底层都是通过代理来实现的

  • 一种是基于JDK的动态代理

  • 一种是基于CgLIB的动态代理

 工作流程

Spring IoC容器的整个工作流程大致可以分为两个阶段:

①、容器启动阶段

容器启动时,会通过某种途径加载 ConfigurationMetaData。除了代码方式比较直接外,在大部分情况下,容器需要依赖某些工具类,比如: BeanDefinitionReader,BeanDefinitionReader会对加载的 ConfigurationMetaData进行解析和分析,并将分析后的信息组装为相应的BeanDefinition,最后把这些保存了bean定义的BeanDefinition,注册到相应的BeanDefinitionRegistry,这样容器的启动工作就完成了。这个阶段主要完成一些准备性工作,更侧重于bean对象管理信息的收集,当然一些验证性或者辅助性的工作也在这一阶段完成。

②、Bean的实例化阶段

经过第一阶段,所有bean定义都通过BeanDefinition的方式注册到BeanDefinitionRegistry中,当某个请求通过容器的getBean方法请求某个对象,或者因为依赖关系容器需要隐式的调用getBean时,就会触发第二阶段的活动:容器会首先检查所请求的对象之前是否已经实例化完成。如果没有,则会根据注册的BeanDefinition所提供的信息实例化被请求对象,并为其注入依赖。当该对象装配完毕后,容器会立即将其返回给请求方法使用。

BeanFactory只是Spring IoC容器的一种实现,如果没有特殊指定,它采用采用延迟初始化策略:只有当访问容器中的某个对象时,才对该对象进行初始化和依赖注入操作。而在实际场景下,我们更多的使用另外一种类型的容器: ApplicationContext,它构建在BeanFactory之上,属于更高级的容器,除了具有BeanFactory的所有能力之外,还提供对事件监听机制以及国际化的支持等。它管理的bean,在容器启动时全部完成初始化和依赖注入操作。

Spring容器扩展机制

先看bean的生命周期

 

 

IoC容器负责管理容器中所有bean的生命周期,而在bean生命周期的不同阶段,Spring提供了不同的扩展点来改变bean的命运。在容器的启动阶段, BeanFactoryPostProcessor允许我们在容器实例化相应对象之前,对注册到容器的BeanDefinition所保存的信息做一些额外的操作,比如修改bean定义的某些属性或者增加其他信息等。

如果要自定义扩展类,通常需要实现 org.springframework.beans.factory.config.BeanFactoryPostProcessor接口,与此同时,因为容器中可能有多个BeanFactoryPostProcessor,可能还需要实现 org.springframework.core.Ordered接口,以保证BeanFactoryPostProcessor按照顺序执行。

postProcessBeforeInitialization()方法与 postProcessAfterInitialization()分别对应图中前置处理和后置处理两个步骤将执行的方法。这两个方法中都传入了bean对象实例的引用,为扩展容器的对象实例化过程提供了很大便利,在这儿几乎可以对传入的实例执行任何操作。注解、AOP等功能的实现均大量使用了 BeanPostProcessor,比如有一个自定义注解,你完全可以实现BeanPostProcessor的接口,在其中判断bean对象的脑袋上是否有该注解,如果有,你可以对这个bean实例执行任何操作。

 

posted @ 2017-11-30 19:16  阿苍老师  阅读(175)  评论(0编辑  收藏  举报