加我微信公众号,一起夯实Java基础,向着诗和远方出发吧~
如果所有的装配工作都交给Spring来自动完成,减少人工的干预,是不是就能减少依赖关系配置带来的麻烦呢?认真做自己的事儿吧,装配交给Spring来!
通过使用Spring的自动装配,我们不再需要去管对象之间复杂的依赖关系,能更加专注于应用核心逻辑的开发。接下来进入个人的free style!各位观众把锅接好了!
一览众山小
- Spring开启组件扫描的方式
- 如何让Bean能够被Spring扫描到
- 怎么让Bean进行自动装配
- 总结
Spring通过组件扫描和自动装配实现自动化装配Bean,通过自动化装配,能最大程度的减少各种显式配置。
Spring开启组件扫描的方式
组件扫描默认是不开启的,因此需要我们手动去启动。Spring开启组件扫描有以下两种方式。
通过Java配置类
- 基本示例
package com.lurker.springaction.yjiang.autowire;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
/**
* 通过@Configuration注解,说明该类是一个配置类,配置信息从这里加载
* 通过@ComponentScan注解,说明开启组件扫描,默认扫描的是该配置类所在包及其子包
*/
@Configuration
@ComponentScan
public class ComponentScanConfig {
}
-
定义组件扫描的基础包
如果我们希望把配置类与核心代码类放在不同的包中,这时候默认的加载地址对我们就没有意义了。这时候我们可以自定义组件扫描的基础包。有以下两种方式:
(1) 指明需要扫描的包的名称
package com.lurker.springaction.yjiang.autowire; import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; @Configuration //指明扫描com.lurker.springaction.yjiang.autowire包 //或者不用value,而使用basePackages,效果是一样的 //value和basePackages都支持以数组的形式配置 //示例:value={"com.lurker.springaction.yjiang.autowire", "..."} @ComponentScan(value = "com.lurker.springaction.yjiang.autowire") public class ComponentScanConfig { }
但是以上方式有个明显的缺陷,其是类型不安全的,如果某次重构代码的时候,包名可能会改变,那么这里设置的
value
或basePackages
就需要改变了。如果你设置了多个多个配置类,想想,多么可怕?那么我们来个更实用的。(2) 指明需要扫描的包中的某个接口
package com.lurker.springaction.yjiang.autowire; import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; //通过basePackageClasses属性,指明需要扫描的包中的接口 //该方式同样支持以数组的形式配置 //示例basePackageClasses={BasePackageLocation.class, ...} @Configuration @ComponentScan(basePackageClasses = BasePackageLocation.class) public class ComponentScanConfig { }
以上方式,用
basePackageClasses
代替了value
和basePackages
属性。程序在解析的时候,会以BasePackageLocation类所在的路径作为基础包路径。我们一般采用第二种方式进行配置,通常的做法是,在需要扫描的包下建立一个空接口,什么都不写。通过标记接口的形式找到基础包,同时由于是空接口,你在重构时可以不加理会,这种方式是对重构友好的。
通过XML文件配置
通过在Spring的xml配置文件中添加如下元素:
<context:component-scan base-package="com.lurker.springaction.yjiang.autowire">
通过这种方式能够取得跟Java配置相同的效果。同时,这种方式也支持同时配置多个包。
如何让Bean能够被Spring扫描到
开启了Spring的组件扫描之后,可以进行扫描指定的包及其子包了。那么,Spring怎么知道哪些才是Bean呢?实际上,Spring的组件是有一个标志的。
package com.lurker.springaction.yjiang.autowire;
import org.springframework.stereotype.Component;
//@Component注解表示该类会作为一个组件类,并告诉Spring要为这个类创建Bean
@Component
public class CDPlayer implements Player {
}
除了@Component
注解外,还有三个注解同样拥有@Component
的作用,他们分别是:
- @Controller:注解Controller层
- @Service:注解Service层
- @Repository:注解DAO层
这三个注解继承自@Component
注解,因此具有相同的功效。在分层架构体系下,在不同的层级运用不同名称的注解,使项目结构更加清晰。
然而注意,这些标注组件的注解,会默认给组件一个名称,默认为类名首字母小写,如其上的Bean名称为cDPlayer
;Java中是允许在不同的包中有相同类名的,这时候怎么区分具有相同类名的Bean呢?Spring的解决方案是:
在@Component
、@Controller
、@Service
、@Repository
这些注解中,填写value
字段,如下:
package com.lurker.springaction.yjiang.autowire;
import org.springframework.stereotype.Component;
//@Component注解表示该类会作为一个组件类,并告诉Spring要为这个类创建Bean
@Component(value="player")
public class CDPlayer implements Player {
}
以上代码,把这个Bean的名称定义为player
,而不再是默认名称cDPlayer
了。除此之外,JDI规范(Java Dependency Injection)还定义了@Named
注解用于为Bean设置ID,其具有与@Component(value="player")
相同的功效。例如:
@Named(value="player")
public class CDPlayer implements Player {
}
如何让Bean实现自动装配
万事俱备,只欠东风。Spring已经扫描到了进行标注后的Bean,并已经将他们放入容器中了,怎么把Bean装配到需要的地方呢?
Spring通过@Autowired
注解来实现Bean的自动装配。@Autowired
注解可以用在类的任何地方。但是,请注意,如果在装配过程中,如果某个需要的Bean不存在,那么Spring会抛出一个异常。如果这个欠缺的Bean不是那么重要,那么你可以尝试通过required
设置为false
来忽略掉它不存在的事实。
- 注解在构造方法上
package com.lurker.springaction.yjiang.autowire;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
@Component
public class CDPlayer {
private Disk disk;
@Autowired
public CDPlayer(Disk disk) {
this.disk = disk;
}
public Disk getDisk() {
return disk;
}
public void setDisk(Disk disk) {
this.disk = disk;
}
}
- 注解在setter方法上
package com.lurker.springaction.yjiang.autowire;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
@Component
public class CDPlayer {
private Disk disk;
public CDPlayer(Disk disk) {
this.disk = disk;
}
public Disk getDisk() {
return disk;
}
@Autowired
public void setDisk(Disk disk) {
this.disk = disk;
}
}
- 注解在属性上
package com.lurker.springaction.yjiang.autowire;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
@Component
public class CDPlayer {
@Autowired
private Disk disk;
public CDPlayer(Disk disk) {
this.disk = disk;
}
public Disk getDisk() {
return disk;
}
public void setDisk(Disk disk) {
this.disk = disk;
}
}
- 注解在任意方法上
package com.lurker.springaction.yjiang.autowire;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
@Component
public class CDPlayer {
@Autowired
public void play(Disk disk) {
System.out.println(disk.toString());
}
}
- 忽视某个Bean不存在的事实
package com.lurker.springaction.yjiang.autowire;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
@Component
public class CDPlayer {
@Autowired(required=false)
public void play(Disk disk) {
System.out.println(disk.toString());
}
}
除此之外,JDI也提供了@Inject
来实现和@Autowired
相同的功能。如果你不想太依赖Spring的注解的话,也可以使用@Inject
注解来实现自动装配。
总结
本文,我们与代码结合,实现了Bean的自动化装配。实际上,这样做我们通过极少的配置就能够让Spring帮我们进行Bean的管理,何乐而不为呢?这种配置方式也是我们主推的。
下一节,我们将一起探讨通过Java配置的方式来进行Bean的装配。一起期待吧!!!
生命不止,学习不休,加油!!!