常用依赖注入注解

1.Map&List注入

在上方我们定义了三个简单的类,People为接口,内部定义了一个返回String 字符串的方法,Student和Teacher实现了这个接口,并重写了这个方法。

1.People

1
2
3
public interface People {
String say();
}

2.Teacher

1
2
3
4
5
6
7
@Service("teacher")
public class Teacher implements People {
@Override
public String say() {
return "我是老师";
}
}

3.Student

1
2
3
4
5
6
7
@Service("student")
public class Student implements People {
@Override
public String say() {
return "我是学生";
}
}

将这两个具体的实现类添加@Service注解交由Spring管理,并且指定具体的Bean名称。

4.PeopleService

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
@Service
public class PeopleService {

private final List<People> peopleList;
private final Map<String, People> peopleMap;

public PeopleService(List<People> peopleList, Map<String, People> peopleMap){
this.peopleList = peopleList;
this.peopleMap = peopleMap;
}

public String say(String key){
People people = peopleMap.get(key);
// People people = peopleList.get(0);
return people.say();
}

}

在PeopleService类中分别使用Map和List进行注入,然后编写具体的业务方法say用来测试。

5.SpringAnnotationApplicationTests

1
2
3
4
5
6
7
8
9
10
11
12
13
@SpringBootTest
class SpringAnnotationApplicationTests {

@Resource
private PeopleService peopleService;

@Test
void contextLoads() {
String say = peopleService.say("teacher");
System.out.println(say);
}

}

在测试类中我们通过Spring获取到PeopleService对象来调用具体方法,使用调试来启动项目。

image-20240830110456327

在代码执行到从map集合中获取Bean对象时,我们发现所有实现了接口的对象都被成功注入到了Map和List集合中,但是由于数据结构的不同,list集合采用了数字作为索引,而map集合采用了bean对象的指定名称作为索引,完成了Bean对象的注入

在实际的开发场景中,Map或者List集合注入可以避免大量的if…else…代码出现,当我们需要某一个服务只需要通过Map集合根据Bean对象的名字来获取对应的服务,使代码保持更加的优美。

2.空注入判断

如果现在Teacher并没有实现接口(也相当于没有提供服务),如果外部调用这个服务的话,肯定会报异常,

这个时候我们可以采用@Autowired(required = false)注解来实现这个场景。

注意去掉Teacher类上的@Service注解(代表Teacher不提供服务)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
@SpringBootTest
class SpringAnnotationApplicationTests {

@Resource
private PeopleService peopleService;
@Autowired(required = false)
private Teacher teacher;
@Test
void contextLoads() {
String say = peopleService.say("student");
// String say = teacher.say();
System.out.println(say);
}

}

image-20240830122418277

我们可以看到teacher对象并未注入,而编译过程也并未报错。

在实际场景中,可能有的服务可能并未提供服务,这时候我们可以采用这个注解来实现不进行注入,避免空指针异常。

3.优先实例化(默认服务)

当一个接口有多个实现类时,我们在注入这个接口时会抛出异常,因为Spring并不知道去实例化哪个类。

Caused by: org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type ‘com.xynu.springannotation.people.People’ available: expected single matching bean but found 2: student,teacher

这时我们就需要指定优先实例化的对象,这里我们采用@Primary注解来配置优先实例化的对象,@Order来处理加载对象的顺序。

记得加上注解

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
@SpringBootTest
class SpringAnnotationApplicationTests {

@Resource
private PeopleService peopleService;
@Autowired(required = false)
private Teacher teacher;
@Resource
private People people;
@Test
void contextLoads() {
// String say = peopleService.say("student");
// String say = teacher.say();
// System.out.println(say);
Class<? extends People> aClass = people.getClass();
System.out.println(aClass);
}

}

image-20240830124137106

这里我们配置Teacher为优先实例化的对象,Student对象作为首先加载的对象,可以看出两者的顺序,但是并不妨碍Teacher优先实例化。

image-20240830124412211

4.检测创建,避免重复

在实际的开发中,我们可能有多种服务来实现一个业务,但是我们可能只需要一个来完成业务,我们可以采用@ConditionalOnMissingBean注解来检测重复的创建,这里我们来添加一个配置类。

1
2
3
4
5
6
7
8
@Configuration
public class PeopleConfig {
@Bean
@ConditionalOnMissingBean(value = People.class, name = "teacher")
public People defaultLogger() {
return new Teacher(); // 默认使用Teacher
}
}

@ConditionalOnMissingBean:

  • value: 指定要检查的 bean 类型。
  • name: 指定要检查的 bean 名称。
  • annotation: 指定要检查的 bean 是否具有某个注解。
  • search: 指定搜索范围 (SEARCH_ALLSEARCH_CHILD)。
  • type: 指定要检查的 bean 类型(等同于 value)。

使用这个注解可以判断服务的创建,并在没有Bean创建时创建默认的Bean对象

注意:强烈建议仅在自动配置类上使用此注解

5.配置是否创建对象

根据配置文件来选择是否创建对象

1
2
3
4
5
6
7
8
9
10
@Configuration
@ConditionalOnProperty(value = "myapp.enableFeature", havingValue = "true", matchIfMissing = false)
public class FeatureConfig {

@Bean
public FeatureService featureService() {
return new FeatureServiceImpl();
}

}

注解参数解释

  • value: 属性名。
  • havingValue: 如果属性值等于此值,则条件成立。
  • matchIfMissing: 如果属性不存在时是否认为条件成立。在上面的例子中设置为 false,表示如果 sdk.config.enabled 属性不存在,则 createTopic 方法不会被激活。

将配置交由外部属性控制,在实际场景中,例如组件开发,需要启动关闭服务,配置业务属性,可以在外部配置文件中修改,而不需要去删除Pom文件中的内容。

6.根据环境配置实例化对象

在开发过程中,可能有些对象的创建与实例化十分的费时,那么我们就可以设置在开发过程中不去实例化这个对象

1
2
3
4
5
6
7
8
9
10
11
@Service("student")
@Order(1)
//根据环境选择是否实例化
@Profile({"prod", "test"})
public class Student implements People {

@Override
public String say() {
return "我是学生";
}
}
1
2
3
4
5
spring:
application:
name: SpringAnnotation
profiles:
active: test

7.引入 Spring 配置

在过去的项目中可能还有使用xml来进行配置Bean对象的工程,我们暂时无法放弃,可以残采用注解来处理这种场景。

1
2
3
4
5
6
7
8
9
10
11
12
13
@SpringBootApplication
@ConfigurationPropertiesScan("com.xynu.springannotation")
@Configurable
@PropertySource("classpath:application.properties")
@ImportResource("classpath:spring.xml")
@EnableScheduling
public class SpringAnnotationApplication {

public static void main(String[] args) {
SpringApplication.run(SpringAnnotationApplication.class, args);
}

}
1
2
3
4
5
6
7
8
9
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd">

<bean id="exampleBean" class="com.xynu.springannotation.entity.test"/>

</beans>

8.其他常用注解

  • @EnableScheduling:用于声明定时任务。
  • @DependsOn({"teacher", "student"}) Bean 对象实例化中,依赖于哪些对象。
  • @Async 异步方法注解,如:@Async public void asyncMethod() {…}
  • 还有许多的注解…