一、Spring装配-自动化装配

@Component和@ComponentScan

通过spring注解(@Component)来表明该类会作为组件类,并告知Spring要为这类创建bean,不过组件扫描默认是不启动的,需要显式的配置Spring,从而命令Spring去寻找带有(@Component)注解的类,并为其创建bean。

1、定义接口

package com.seven.springTest.service;public interface HelloWorldApi {public void sayHello();
}

2、定义接口的实现类

package com.seven.springTest.service.impl;import org.springframework.stereotype.Component;import com.seven.springTest.service.HelloWorldApi;@Component    //通过注解指定该类组件类,告知spring要为它创建Bean
public class PersonHelloWorld implements HelloWorldApi {@Overridepublic void sayHello() {System.out.println("Hello World,This Is Person!");}
}

3、前面说过了,spring并不能自动启用组件扫描,需要进行显式的配置,这里通过java类来进行显式的配置,定义java配置类HelloWorldConfig,在配置类中我们没有显式的声明任何bean,只不过是使用了@CompontentScan注解来启用组件扫描

package com.seven.springTest;import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;@Configuration
@ComponentScan   // 启用组件扫描
public class HelloWorldConfig {}

现在所有的工作已经完成,我们来测试下

package com.seven.springTest.main;import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import com.seven.springTest.HelloWorldConfig;
import com.seven.springTest.service.HelloWorldApi;public class HelloWorldTest {public static void main(String[] args) {//1. 声明Spring上下文,采用java配置类ApplicationContext ac = new AnnotationConfigApplicationContext(HelloWorldConfig.class);//2. 声明Spring应用上下文,采用xml配置//ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");//通过Spring上下文获取Bean,在这里Spring通过自动扫描发现了PersonHelloWorld的实现,并自动创建bean。HelloWorldApi hwapi = ac.getBean(HelloWorldApi.class);//通过sayHello()的输入内容可以看到,hwapi为PersonHelloWorld的实例
        hwapi.sayHello();}
}

通过上的代码,在控制台会输出下面的内容 “Hello World,This Is Person!”

Spring容器通过组件扫描发现了PersonHelloWorld类,并为它创建了对应的bean。到此为止,我们通过简单的注解实现自动化装配,在上面的案例中,HelloWorldConfig配置类@ComponentSan如果没有其他配置,只会扫描HelloWorldConfig所在包或者它的子包,如果需要制定扫描的包,可以通过

@ComponentScan("com.seven.springTest")

或者

@ComponentScan(basePackages="com.seven.springTest")

basePackages允许设置多个包,,只需要把basePackages熟悉设置成一个数组即可

@ComponentScan(basePackages={"com.seven.springTest.service","com.seven.springTest.impl"})

除了通过java配置类来设置Spring启用组件扫描,还可能通过xml类显式配置,参考下面xml配置,并在获取Spring应用上下文时通过xml来初始化。

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx"xmlns:util="http://www.springframework.org/schema/util" xmlns:context="http://www.springframework.org/schema/context"xsi:schemaLocation="http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.2.xsdhttp://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsdhttp://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.2.xsdhttp://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.2.xsd"><!-- 启用Spring组件扫描--><context:component-scan base-package="com.seven.springTest"></context:component-scan></beans>

在上面的案例中,我们通过@Component和@ComponentScan来隐式的配置完成了Bean的装配工作,接下来我们深入的探讨下@Component和@ComponentScan

Spring容器在管理Bean的时候,会给每一个Bean有一个ID标识,上面的例子,如果HelloWorldApi的实现类有多个,那么Spring容器该怎么分配Bean呢,如果我们在使用@Component的时候,没有明确的给PersonHelloWorld设置一个ID,Spring容器会默认给bean给定一个ID,一般为类名(第一个字母会变为小写,例如跑personHelloWorld),所以下面的代码也是成立的

//通过bean的ID来获取实例
HelloWorldApi hwapi = (HelloWorldApi) ac.getBean("personHelloWorld");
hwapi.sayHello();

同时我们也可以为bean设置ID,如下:

@Component("person")    //为bean设置ID为“person”
public class PersonHelloWorld implements HelloWorldApi {@Overridepublic void sayHello() {// TODO Auto-generated method stubSystem.out.println("Hello World,This Is Person!");}}

这样我们在获取bean的时候就可通过ID来获取,如下:

// 根据设置的bean ID来获取bean
HelloWorldApi hwapi = (HelloWorldApi) ac.getBean("person");

在以上的案例中我们使用了@Component和@ComponentScan实现了组件扫描,目前为止我们都是对单一的对象进行操作,如果程序复杂点,对象之间存在依赖,该如何处理呢?下面我们就来研究下如何为bean添加注解实现自动装配

@AutoWired

在上面的案例中Person对整个世界说了一句Hello,可说话只有旁边的人知道,我们需要让更多的听到我们的“hello world”,我们就需要一些工具,我们使用电视来广播就能让更多的人听到了,首先我们定义一个传播工具接口

package com.seven.springTest.service;public interface TransmittingTool {void work(String message);
}

接下来我们来创建我们的TV

package com.seven.springTest.service.impl;
import org.springframework.stereotype.Component;import com.seven.springTest.service.TransmittingTool;@Component   //设置为需要被扫描到的组件
public class TVTool implements TransmittingTool {@Overridepublic void work(String message) {//传播工具工作,把我们的消息传播出去
        System.out.println(message);}
}

接下来我们需要对我们之前的PersonHelloWorld的sayHello()方法进行一些修改,让它可以通过传播工具来对全世界说Hello

package com.seven.springTest.service.impl;import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;import com.seven.springTest.service.HelloWorldApi;
import com.seven.springTest.service.TransmittingTool;@Component
public class PersonHelloWorld implements HelloWorldApi {//定义传播工具@Autowired   //1.直接变量添加注解private TransmittingTool transmittingTool;//  @Autowired   //2. 通过构造函数注入依赖
//  public PersonHelloWorld(TransmittingTool transmittingTool) {
//      this.transmittingTool = transmittingTool;
//  }//  @Autowired    //3. 通过属性的Setter方法注入依赖
//  public void setTransmittingTool(TransmittingTool transmittingTool) {
//      this.transmittingTool = transmittingTool;
//  }//  @Autowired  //4. 通过其他函数注入依赖
//  public void inserttool(TransmittingTool transmittingTool){
//      this.transmittingTool=transmittingTool;
//  }
    @Overridepublic void sayHello() {// 通过传播工具来sayHellotransmittingTool.work("Hello World,This Is Person!--form TV");}
}

首先我们定义了一个传播工具,这个工具的具体实现我们不清楚,需要Spring容器去给我注入依赖。
@Autowired直接可以使用在类变量、构造函数、Setter和其他任何方法上,参考代码中1-4的实现

  1. 直接在变量上添加注解
  2. 在构造函数上添加注解,在spring容器通过构造器实例化bean的时候,会传入一个提供给transmittingTool的实例,注入依赖;
  3. 通过Setter方法或者其他函数,Spring在初始化bean以后,会尽量的去满足bean的所有依赖,如果使用第4个种注入,我们在HelloWorldTest中重来没有调用过inserttool()方法,可是sayHello()还是能正常执行,Spring会去根据@Autowired来尽量尝试去注入PersonHelloWorld的依赖。

如果能够配置到1个满足要求的依赖,那么这个被依赖的bean就会被装配进来,如果没有匹配的依赖bean,那么应用上下文创建的时候,Spring会抛出一个异常,为了避免异常,我们可以把@Autowired的required设置为false;

@Autowired(required=false)   //2. 通过构造函数注入依赖
public PersonHelloWorld(TransmittingTool transmittingTool) {this.transmittingTool = transmittingTool;
}

@Autowired的required设置给false后,Spring为尝试给bean自动装配,注入依赖,如果没有匹配的bean的话,Spring将会让这个bean处于未装配的状态,当时把required设置为false的时需要注意,因为这个依赖bean处于未装配状态,在调用依赖的时候,如果你的代码做null的检查,这个处于未装配状态的属性有可能会发生异常。

如果有多个bean能满足依赖关系的话,Spring也会抛出异常,表明没有明确指出选择哪个bean进行自动装配。这个在后面我会单独开一篇讲解Spring的高级装配,到时候在详细说明,大家可以关注后续的文章。

转载于:https://www.cnblogs.com/fhyl/p/6971382.html

Spring学习系列(二) 自动化装配Bean相关推荐

  1. Spring实战之二:装配Bean

    2.1 Spring配置的可选方案 Spring提供了三种装配机制: 在XML中显式配置 在Java中显式配置 隐式的bean发现机制和自动装配 Best Practice:尽可能使用自动配置的机制, ...

  2. Spring学习总结二

    Spring框架的代理与AOP.AspectJ Spring学习总结二 0.在理解什么是AOP之前的一些话 1.什么是AOP 2.AOP的重要概念 3.代理模式 3.1.静态代理 3.2.动态代理 3 ...

  3. 第2章—装配Bean—自动化装配Bean

    自动化装配Bean 2.1.Spring配置可选方案 ​ 装配是依赖注入DI的本质,Spring提供了以下三种注入的装配机制: 在XMl中进行显式配置 在java中进行显式配置 隐式的Bean发现机制 ...

  4. 图机器学习(GML)图神经网络(GNN)原理和代码实现(前置学习系列二)

    图机器学习(GML)&图神经网络(GNN)原理和代码实现(PGL)[前置学习系列二] 上一个项目对图相关基础知识进行了详细讲述,下面进图GML networkx :NetworkX 是一个 P ...

  5. Linux学习系列二:Linux中的常用命令

    这个系列的Linux教程主要参考刘遄老师的<Linux就该这么学>.用的系统是RHEL8,如果遇见一些命令出现问题,请首先检查自己的系统是否一致,如果不一致,可网上查一下系统间某些命令之间 ...

  6. Pyside2 学习系列二:PyInstaller打包项目exe (超详细的Pyside2 攻略)

    继上一篇文章创建了项目后,本章我们进行项目的打包工作. 本项目的所有演示代码:github可在这里下载. 打包只用的工具为PyInstaller. 打包步骤 1 准备环境 1.1 安装`PyInsta ...

  7. A.图机器学习(GML)图神经网络(GNN)原理和代码实现(前置学习系列二)

    图学习图神经网络算法专栏简介:主要实现图游走模型(DeepWalk.node2vec):图神经网络算法(GCN.GAT.GraphSage),部分进阶 GNN 模型(UniMP标签传播.ERNIESa ...

  8. Spring入门(二):自动化装配bean

    Spring从两个角度来实现自动化装配: 组件扫描(component scanning):Spring会自动发现应用上下文中需要创建的bean. 自动装配(autowiring):Spring会自动 ...

  9. Spring学习之Spring三种装配机制:(一)自动化装配bean

    装配:创建应用组件(对象)之间的协作的行为,这也是Spring依赖注入(DI)的本质. Spring提供了三种装配机制: 隐式的自动装配bean: 通过java代码装配bean(显示): 通XML中装 ...

最新文章

  1. db2 linux 数据导出_linux db2 导出数据库
  2. nginx conf 文件
  3. 31.return和call的区别
  4. 百度地图SDK v3.6.0以上版本找不到PoiOverlay类的解决方法
  5. java时序图工具_基于时序数据库,实现百亿级话单实时全景监控分析
  6. JFreeChart(八)之时序图
  7. spring mail 发送html simple,SpringBoot整合Mail邮件发送
  8. React Native集成Redux框架讲解与应用
  9. c语言 汇编中断程序,ARM7 C语言嵌入汇编开关中断
  10. python requests 异步调用_构建高效的python requests长连接池详解
  11. chrome最强大的浏览器插件推荐,只要你会用其他的插件你可以删除了
  12. 纯前端导出PDF分页截取问题处理
  13. filesplit的Python项目详细描述
  14. 灵雀云Kube-OVN:基于OVN的开源Kubernetes网络实践
  15. VS code Markdown Preview Enhanced 预览白色改为黑色
  16. 波司登 × 阿里云 × 奇点云:教科书式的数智化转型实践
  17. 没有性生活、觉得老板傻……我们问了2000个人辞职的理由
  18. Catalan 数的应用
  19. WORDPRESS QQ扫码登录插件
  20. mysql企业备份工具(MEB)之mysqlbackup安装及使用

热门文章

  1. strust2自定义interceptor的基本方法及操作
  2. Swift 中使用 SQLite——新增数据
  3. Swift基础 - - 高德地图实践
  4. 完爆Facebook/GraphQL,APIJSON全方位对比解析(一)-基础功能
  5. 特斯拉解锁对汽车电池容量的软件限制,以帮助用户逃离飓风危险
  6. 中小企业对于云计算的3大误解
  7. 生鲜电商APP开发,有哪五大商机?
  8. Xamarin SimplerCursorAdapter 适配器(三)
  9. My deep learning reading list
  10. 【VMCloud云平台】SCO(四)流程准备