Spring中的IoC容器

什么是IoC

什么是耦合和内聚

  • 耦合指的就是模块之间的依赖关系。模块间的依赖越多,则表示耦合度越高,相应的维护成本就越高。
  • 内聚指的是模块内功能之间的联系。模块内功能的联系越紧密,则表示内聚度越高,模块的职责也就越单一。

因此在程序开发中应该尽可能的下降耦合,提升内聚。也就是设计原则中的开闭原则和单一职责原则。java

工厂模式

工厂模式就是用来解决程序间耦合的一种设计模式。能够把全部要建立的对象放在工厂的一个集合里,当须要使用这个对象的时候,直接从工厂里面取出来用就行。
工厂模式的优势:spring

  • 一个调用者想建立一个对象,只须要指定相应的名字便可从工厂中得到这个对象。
  • 屏蔽了产品的具体实现,调用者只关心产品的接口。

控制反转(IoC)

控制反转在维基百科中的定义:数据库

控制反转(Inversion of Control,缩写为IoC),是面向对象编程中的一种设计原则,能够用来减低计算机代码之间的耦合度。其中最多见的方式叫作依赖注入(Dependency Injection,简称DI),还有一种方式叫“依赖查找”(Dependency Lookup)。经过控制反转,对象在被建立的时候,由一个调控系统内全部对象的外界实体,将其所依赖的对象的引用传递(注入)给它。编程

下面再从控制和反转两个词分两个方面来理解:设计模式

  • 谁控制谁?IoC容器控制了对象。控制什么?控制了对象要获取的外部资源(其它对象或数据等)
  • 什么是反转?是IoC容器查找并注入依赖给对象,对象是被动的接受,而不是主动的建立,因此是反转。

经过new方式来主动获取对象:
springboot

image.png

经过IoC容器获取对象(注意看箭头的方向,是否是反转了):
bash

\[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-McaurwWV-1576318447166)(https://i.loli.net/2019/12/14/U1o3cR6mps4kQvH.png)\]

有了IoC容器后,把建立和查找依赖对象的控制权交给了容器,由容器进行注入组合对象,因此对象与对象之间是松散耦合,这样也方便测试,利于功能复用,更重要的是使得程序的整个体系结构变得很是灵活。session

依赖注入(DI)

依赖注入在维基百科中的定义函数

在软件工程中,依赖注入是种实现控制反转用于解决依赖性设计模式。一个依赖关系指的是可被利用的一种对象(即服务提供端) 。依赖注入是将所依赖的传递给将使用的从属对象(即客户端)。该服务是将会变成客户端的状态的一部分。 传递服务给客户端,而非容许客户端来创建或寻找服务,是本设计模式的基本要求。测试

其实依赖注入和控制反转表达的是一个意思。控制反转是一种思想,而依赖注入是这个思想的最典型的实现方法。
由IoC来控制对象的依赖,经过构造函数、变量或Setter等方法来将依赖注入到对象中,这样就将对象和对象的依赖进行了解耦。

spring中的工厂类

咱们经常使用的spring容器是ApplicationContext,先来看一下它的依赖结构。

\[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Y7hwa6zl-1576318447168)(https://i.loli.net/2019/12/14/5X8F4lUgvemJMTc.png)\]
由图可知,spring容器中的顶层接口是BeanFactory。ApplicationContext是它的子接口(注意这个也是个接口哦)。它默认一读取配置文件,就会建立对象放到容器中。
再来看一下ApplicationContext的三个主要的实现类。
\[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-R9nohp4k-1576318447169)(https://i.loli.net/2019/12/14/hsiDTJt4YxOHWn8.png)\]

  • ClassPathXmlApplication:它是从类的根路径下加载xml配置文件(推荐用这种)。
  • FileSystemXmlApplication: 它是从磁盘路径上加载配置文件,配置文件能够在磁盘的任意位置。(但使用不灵活,不推荐)
  • AnnotationConfigApplication:当咱们使用注解配置容器对象时,须要使用此类来建立spring容器。它用来读取注解。(springboot默认使用这个)

Bean的建立和管理

bean标签

做用 : 用于配置对象让spring来建立的。默认状况下它调用的是类中的无参构造函数。若是没有无参构造函数则不能建立成功。
属性

  • id:给对象在容器中提供一一个惟一 标识。用于获取对象。
  • class:指定类的全限定类名。用于反射建立对象。默认状况下调用无参构造函数。
  • scope:指定对象的做用范围。
    • singleton : 单例对象,也是默认的。
    • prototype : 多例对象,每次都建立一个不一样的对象。
    • request :WEB 项目中,Spring建立一个Bean的对象,将对象存入到request域中.
    • session : WEB项目中,Spring建立一个Bean的对象,将对象存入到session域中。
    • global session:WEB项目中,应用在集群环境.若是没有集群环境那么globalSession至关于session.
  • init-method:指定类中的初始化方法名称。
  • destroy-method:指定类中销毁方法名称。

bean的做用范围和生命周期

  1. 单例对象: scope="singleton" 一个应用只有一一个对象的实例。它的做用范围就是整个引用。 生命周期:
  • 对象出生:当应用加载,建立容器时,对象就被建立了。
  • 对象活着:只要容器在,对象-直活着。
  • 对象死亡:当应用卸载,销毁容器时,对象就被销毁了。
  1. 多例对象: scope="prototype" 每次访问对象时,都会从新建立对象实例。 生命周期:
  • 每次访问对象时,都会从新建立对象实例。
  • 对象活着:只要对象在使用中,就一直活着。
  • 对象死亡:由java的垃圾回收器机制来处理。

spring中的注解

用于建立对象的

至关于<bean id = "" class = "" />
复制代码

1 、@component
做用:把资源让spring来管理,至关于在xml中注册一个bean。
属性:value:指定bean的id.若是不指定value属性,默认bean的id是当前类的类名。首字母小写。 二、 @Service 、@Repository 、@Controller 他们都是对@Component注解的衍生,其实做用是如出一辙的,只是提供了更明确的语义化。

  • @Repository:通常用于持久层的注解。
  • @Service:通常用于业务层的注解
  • @Controller:通常用于表现层的注解。

用于注入数据的

至关于<property name = "" ref = "" />  或
      <property name= "" value = "" />
复制代码

@Autowried

做用:自动按照类型注入。当使用注解注入属性时,set 方法能够省略。它只能注入其余bean类型(ByType)。当有多个类型匹配时,使用要注入的对象变量名称做为bean的id (ByName) ,在spring容器查找,找到了也能够注入成功。找不到就报错。
三种注入方式(属性输入、Setter注入、构造函数注入):

/** * 使用变量注入依赖 */
@Autowired
private IAccountDao accountDao ;

/** * 使用构造器注入 spring推荐使用这个 * @param accountDao 要注入的依赖 */
@Autowired
public AccountServiceImpl(IAccountDao accountDao) {
    this.accountDao = accountDao;
}

/** * 使用Setter注入 * @param accountDao 要注入的依赖 */
@Autowired
public void setAccountDao(IAccountDao accountDao) {
    this.accountDao = accountDao;
}
复制代码

@Qualifier

做用:在自动按照类型注入的基础之上,再按照Bean的id注入。它在给字段注入时不能独立使用,必须和@Autowire一块儿使用;可是给方法参数注入时,能够独立使用。
属性:value:指定bean的id.

@Resource

做用:@Resource采用 name 属性。默认状况下,Spring 将 value 解释为要注入的 bean name。也就是ByName注入。
属性:value:指定bean的id.

@Value

做用:用于注入基本类型数据和String类型数据
属性:用于指定值,可以使用SpEL表达式。

用于改变做用域的

至关于<bean id = "" class = "" scope = ""/>中的scope属性
复制代码

@Scope

做用:指定bean的做用范围。
属性:value:指定bean的做用范围。取值: singleton prototype request session globalsession。

生命周期相关的

至关于<bean id = "" class = "" init-method = "" destory-method = ""/>中的init-method属性和destory-method属性
复制代码

@PostConstruct

做用:用于指定初始化方法

@PostDestory

做用:用于指定销毁方法

spring中的新注解

@Configuration

做用:用于指定当前类是一个spring配置类,当建立容器时会从该类上加载注解。获取容器时须要使用AnnotationApplicationContext (有@Configuration注解的类. class)。
属性:value:用于指定配置类的字节码

@ComponentScan

做用:用于指定spring在初始化容器时要扫描的包。做用和在spring的xml配置文件中的<context : component-scan base-package="com. itheima"/>是同样的。
属性:basePackages: 用于指定要扫描的包。和该注解中的value属性做用同样。

Bean

做用:该注解只能写在方法(该方法的返回值做为bean放到容器中)上,代表使用此方法建立一个对象, 而且放入spring容器。
属性:name:给当前@Bean注解方法建立的对象指定一个名称 (即bean的id)。

@PropertySource

做用:用于加载.properties文件中的配置。例如咱们配置数据源时,能够把链接数据库的信息写到properties配置文件中,就可使用此注解指定properties配置文件的位置。
属性:value[] :用于指定properties文件位置。若是是在类路径下,须要写上classpath:。
示例:

@Configuration
@PropertySource("classpath:jdbc.properties")
public class JdbcConfig{
}
复制代码

@Import

做用:用于导入其余配置类,在引入其余配置类时,能够不用再写@Configuration注解。固然,写上也没问题。
属性:value[] :用干指定其余配置类的字节码。
示例:

@Configuration
@ComponentScan(basePackages = "com.ncusoft.springDemos")
@Import({ JdbcConfig.class })
public class SpringConfiguration {
}
复制代码

选择XML仍是注解?

先来看一下各自的优点:

  • 注解的优点:配置简单,维护方便(咱们找到类,就至关于找到了对应的配置)。
  • XML的优点:修改时,不用改源码。不涉及从新编译和部署。

应该根据实际的开发来选择使用(springboot推荐使用注解),通常在source code(源代码)中的类使用注解来建立bean(更方便,只需一个注解搞定)。从外部引入的依赖可选择使用XML来建立bean。

相关文章
相关标签/搜索