第1章:Spring 基础
spring 简史
- xml配置
spring1.x,主要配置各种Bean - 注解配置
spring2.x,基本配置用xml,业务配置用注解 - Java 配置
spring3.x,spring boot 都推荐使用 Java 配置
spring 概述
包括内容很多,每一块都能单独拿出来讲。
spring 在 idea 中的搭建
为了体验spring,所以建立的是一个 maven quick start 的项目,建立后的 pom.xml 配置如下:
4.0.0
com.wisely
highlight_spring5_idea
0.0.1-SNAPSHOT
jar
highlight_spring5_idea
http://maven.apache.org
UTF-8
1.8
org.springframework
spring-context
5.0.1.RELEASE
junit
junit
3.8.1
test
org.apache.maven.plugins
maven-compiler-plugin
3.7.0
${java.version}
spring 基础配置
spring 框架四大原则:
- 使用 POJO(Plain Old Java Object) 进行轻量级和最小侵入式开发
- 依靠依赖注入和接口实现松耦合
- 使用 AOP 和默认习惯进行声明式编程
- 使用 AOP 和模板减少代码冗余
依赖注入
依赖注入是通过配置+容器的方式实现的,配置可以通过 xml 配置、注解配置、java 配置实现,这些配置称为元配置数据,这些元配置本身不会有运行的能力,是通过程序解析代码后,根据元数据来做相应操作。
先介绍一个概念 Beans,Beans 是被 spring 容器管理的 POJO,Beans 以及 Beans 之间的依赖关系都是通过元数据来配置的,而这些元数据的使用方则是称为 IoC container 的容器。
Spring Framework 中 IoC container 主要代码是在 org.springframework.beans
和 org.springframework.context
两个包中。
在前面设置maven的依赖的时候,我们导入了 context, 看下依赖图:
可以看到org.springframework.context
本身依赖于org.springframework.beans
IoC 容器具体到具体的类是BeanFactory
,里面定义了基本的接口来获取 Beans,而ApplicationContext
是为了企业级应用而对BeanFactory
的一个扩展,具体可以看依赖图:
接口org.springframework.context.ApplicationContext
的职责是负责Beans的实例化,配置,组装等工作,而这些对Beans的具体操作都是通过配置数据来控制的。因为ApplicationContext
是一个接口,只要实现这个接口,就可以作为一个容器来使用,常用的容器有ClassPathXmlApplicationContext
和FileSystemXmlApplicationContext
。
下面是一个ApplicationContext
工作的概括图:
ApplicationContext
通过我们配置的元数据,棒我们实例化、配置好我们需要的Beans,当具体的一个
ApplicationContext
对象生成结束的时候,我们的系统也就做好运行的准备了。
元配置数据
配置有3类方式:xml、注解和java,我们此处主要讲xml。一个典型的xml配置文件结构如下:
id 是对Bean的唯一标识,class则是具体的类名。
容器实例化
可以通过下面的代码得到一个具体的容器:
ApplicationContext context = new ClassPathXmlApplicationContext("services.xml", "daos.xml");
我们可以看到对于Beans的声明我们可以放到多个不同的xml文件中,还有另一个方式是在一个xml中导入其他的xml文件,如下:
Bean 概览
IoC 容器管理的Beans都是通过xml中BeanDefinition
对象,包含的信息有:
- bean 具体的实现类
- bean 在 IoC 中的行为(scope, lifecycle callbacks 等)
- bean 的依赖
- bean 的其他配置信息
下面是一个具体属性和作用:
name
一个Bean可以有多个名字,也可以没有名字,在xml中可以通过id or name
来指定,id只能指定唯一一个名字,name可以通过","或者";"或者空格将多个名字隔开,另一种指定名字的方式是通过alias,如下:
如果我们没有指定id或者name,IoC容器会自动生成一个名字。
对于不指定名字的使用场景是:inner beans 和 autowiring collaborators,后面介绍。
实例化
实例化一个Bean的方式有两种
- 通过反射获取构造函数
- 静态工厂
先看第一种,构造函数的方式。
构造函数的方式一般要求Bean有个默认的构造函数(没有任何参数),然后通过set方式来设置Bean的属性,配置如下:
如果构造函数需要一些依赖注入,稍后介绍怎么完成。
另外一种方式是静态工厂,看配置:
public class ClientService {
private static ClientService clientService = new ClientService();
private ClientService() {}
public static ClientService createInstance() {
return clientService;
}
}
另一种工厂是来自于另一个Bean的方法
public class DefaultServiceLocator {
private static ClientService clientService = new ClientServiceImpl();
public ClientService createClientServiceInstance() {
return clientService;
}
}
依赖关系
依赖注入的种类
常见的依赖注入有构造函数注入和set方法注入,先来看构造函数注入。
构造函数注入
假设我们有下面的类:
package x.y;
public class Foo {
public Foo(Bar bar, Baz baz) {
// ...
}
}
我们可以配置下面的xml,通过标签
来指定:
另外一种是构造函数依赖的是普通的类型,如int,String等,看例子:
package examples;
public class ExampleBean {
// Number of years to calculate the Ultimate Answer
private int years;
// The Answer to Life, the Universe, and Everything
private String ultimateAnswer;
public ExampleBean(int years, String ultimateAnswer) {
this.years = years;
this.ultimateAnswer = ultimateAnswer;
}
}
此时参数不再是Bean了,我们通过type,value
的方式实现:
也可以通过index-value的方式:
另外一种是在开启debug模式下使用的name-value,
当然不想开启debug,也可以通过java的注释@ConstructorProperties
来做:
package examples;
public class ExampleBean {
// Fields omitted
@ConstructorProperties({"years", "ultimateAnswer"})
public ExampleBean(int years, String ultimateAnswer) {
this.years = years;
this.ultimateAnswer = ultimateAnswer;
}
}
下面总结下依赖解析的整个过程:
-
ApplicationContext
创建并且读取元配置信息生成BeanDefinition
对象 - 对于每个Bean,其依赖通过
properties,constructor arguments ,static-factory method
来声明的。 - 每个
properties,constructor arguments
是以值的形式或者对其他Bean的引用提供的。 - 每个
properties,constructor arguments
如果是值的话,能自动转换到对应的类型(int,long,boolean等)。
下一篇将会详细介绍下依赖。
更好的阅读体验可以看 https://www.zybuluo.com/zhuanxu/note/943431