首先,我们将探讨一些Spring
框架中IOC
(Inversion of Control
)的高级特性,特别是组件扫描的相关知识。组件扫描是Spring
框架中一个重要的特性,它可以自动检测并实例化带有特定注解(如@Component
, @Service
, @Controller
等)的类,并将它们注册为Spring
上下文中的bean
。这里,我们会通过一些详细的例子来阐明这些概念,并且展示如何在实际的代码中使用这些特性。
@ComponentScan
注解是用于指定Spring
在启动时需要扫描的包路径,从而自动发现并注册组件。
我们设置组件扫描路径包括两种方式:
@ComponentScan("com.nowjava.demo")
,等同于@ComponentScan(basePackages = {"com.nowjava.demo"})
,Spring
会扫描指定包下的所有类,并查找其中带有@Component
、@Service
、@Repository
等注解的组件,然后将这些组件注册为Spring
容器的bean
。@ComponentScan(basePackageClasses = {ExampleService.class})
,Spring
会扫描ExampleService
类所在的包以及其所有子包。接下来,给出了一个完整的例子,说明如何使用第二种方式来设置组件扫描路径。这可以通过设置@ComponentScan的basePackageClasses
属性来实现。例如:
@Configuration
@ComponentScan(basePackageClasses = {ExampleService.class})
public class BasePackageClassConfiguration {
}
以上代码表示,Spring
会扫描ExampleService
类所在的包以及其所有子包。
全部代码如下:
首先,我们创建一个名为ExampleService
的服务类
package com.nowjava.demo.service;
import org.springframework.stereotype.Service;
@Service
public class ExampleService {
}
接着在bean
目录下创建DemoDao
package com.nowjava.demo.bean;
import org.springframework.stereotype.Component;
@Component
public class DemoDao {
}
然后在配置类中设置组件扫描路径
package com.nowjava.demo.configuration;
import com.nowjava.demo.service.ExampleService;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
@Configuration
@ComponentScan(basePackageClasses = ExampleService.class)
public class BasePackageClassConfiguration {
}
我们还会创建一个名为DemoApplication
的类,这个类的作用是启动Spring
上下文并打印所有注册的bean
的名称。
package com.nowjava.demo;
import com.nowjava.demo.configuration.BasePackageClassConfiguration;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import java.util.Arrays;
public class DemoApplication {
public static void main(String[] args) {
AnnotationConfigApplicationContext ctx = new
AnnotationConfigApplicationContext(BasePackageClassConfiguration.class);
String[] beanDefinitionNames = ctx.getBeanDefinitionNames();
Arrays.stream(beanDefinitionNames).forEach(System.out::println);
}
}
运行上述DemoApplication
类的main
方法,就会在控制台上看到所有注册的bean
的名称,包括我们刚刚创建的ExampleService
。
现在,如果我们在ExampleService
类所在的包或者其任意子包下创建一个新的类(例如TestService
),那么这个组件类也会被自动注册为一个bean
。这就是basePackageClasses
属性的作用:它告诉Spring
要扫描哪些包以及其子包。
package com.nowjava.demo.service;
import org.springframework.stereotype.Service;
@Service
public class TestService {
}
如果再次运行DemoApplication
类的main
方法,就会看到TestService
也被打印出来,说明它也被成功注册为了一个bean
。
我们可以看到这个DemoDao
始终没有被扫描到,我们看一下@ComponentScan
注解的源码
可以看到basePackageClasses
属性这是一个数组类型的,有人会疑问了,刚刚我们写的@ComponentScan(basePackageClasses = ExampleService.class)
,这没有用到数组啊,为什么这里还能正常运行呢?
如果数组只包含一个元素,可以在赋值时省略数组的大括号 {}
,这是Java
的一种语法糖。在这种情况下,编译器会自动把该元素包装成一个数组。
例如,以下两种写法是等价的:
@ComponentScan(basePackageClasses = {ExampleService.class})
和
@ComponentScan(basePackageClasses = ExampleService.class)
在上面两种情况下,ExampleService.class
都会被包装成一个只包含一个元素的数组。这是Java
语法的一个便利特性,使得代码在只有一个元素的情况下看起来更加简洁。
那么为了DemoDao
组件被扫描到,我们可以在basePackageClasses
属性加上DemoDao
类,这样就可以扫描DemoDao
组件所在的包以及它的子包。
package com.nowjava.demo.configuration;
import com.nowjava.demo.bean.DemoDao;
import com.nowjava.demo.service.ExampleService;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
@Configuration
@ComponentScan(basePackageClasses = {ExampleService.class, DemoDao.class})
public class BasePackageClassConfiguration {
}
运行结果
@ComponentScan
注解的源码还有不少,后面我们用到再说
除了基本的包路径扫描,Spring
还提供了过滤功能,允许我们通过设定过滤规则,只包含或排除带有特定注解的类。这个过滤功能对于大型项目中的模块划分非常有用,可以精细控制Spring
的组件扫描范围,优化项目启动速度。
在Spring
中可以通过@ComponentScan
的includeFilters
属性来实现注解包含过滤,只包含带有特定注解的类。
在下面这个例子中,我们将创建一些带有特定注解的组件,并设置一个配置类来扫描它们。
全部代码如下:
创建一个新的注解Species
:
package com.nowjava.demo.annotation;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface Species {
}
接下来,我们将创建三个不同的组件,其中两个包含Species
注解:
package com.nowjava.demo.bean;
import com.nowjava.demo.annotation.Species;
import org.springframework.stereotype.Component;
@Species
public class Elephant {
}
Elephant
类被@Species
修饰,没有@Component
修饰。
package com.nowjava.demo.bean;
import org.springframework.stereotype.Component;
@Component
public class Monkey {
}
Monkey
只被@Component
修饰
package com.nowjava.demo.bean;
import com.nowjava.demo.annotation.Species;
import org.springframework.stereotype.Component;
@Component
@Species
public class Tiger {
}
如上所示,Tiger
有@Component
和@Species
修饰。
接着,我们需要创建一个配置类,用于扫描和包含有Species
注解的组件:
package com.nowjava.demo.configuration;
import com.nowjava.demo.annotation.Species;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.ComponentScan.Filter;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.FilterType;
@Configuration
@ComponentScan(basePackages = "com.nowjava.demo",
includeFilters = @Filter(type = FilterType.ANNOTATION, classes = Species.class),
useDefaultFilters = false)
public class FilterConfiguration {
}
在这个配置类中,我们设置了 @ComponentScan
注解的 includeFilters
属性,FilterType.ANNOTATION
代表按注解过滤,这里用于扫描包含所有带有 Species
注解的组件。注意,useDefaultFilters = false
表示禁用了默认的过滤规则,只会包含标记为 Species
的组件。
有人可能会疑问了,useDefaultFilters = false
表示禁用了默认的过滤规则,什么是默认的过滤规则?
在Spring
中,当使用@ComponentScan
注解进行组件扫描时,Spring
提供了默认的过滤规则。这些默认规则包括以下几种类型的注解:
默认不写useDefaultFilters
属性的情况下,useDefaultFilters
属性的值为true
,Spring
在进行组件扫描时会默认包含以上注解标记的组件,如果将useDefaultFilters
设置为false
,Spring
就只会扫描明确指定过滤规则的组件,不再包括以上默认规则的组件。所以,useDefaultFilters = false
是在告诉Spring
我们只想要自定义组件扫描规则。
最后,我们创建一个主程序,来实例化应用上下文并列出所有的Bean
名称:
package com.nowjava.demo;
import com.nowjava.demo.configuration.FilterConfiguration;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
public class DemoApplication {
public static void main(String[] args) {
ApplicationContext ctx = new AnnotationConfigApplicationContext(FilterConfiguration.class);
String[] beanNames = ctx.getBeanDefinitionNames();
for (String beanName : beanNames) {
System.out.println(beanName);
}
}
}
当我们运行这个程序时,会看到输出的Bean
名字只包括被Species
注解标记的类。在这个例子中会看到 Tiger
和 Elephant
,但不会看到 Monkey
,因为我们的配置只包含了被 Species
注解的类。
运行结果:
如果useDefaultFilters
属性设置为true
,那么运行程序时输出的Bean
名字将会包括Monkey
。
总结:上面介绍了如何使用Spring
的@ComponentScan
注解中的includeFilters
属性和useDefaultFilters
属性来过滤并扫描带有特定注解的类。通过自定义注解和在配置类中设置相关属性,可以精确地控制哪些类被Spring
容器扫描和管理。如果设置useDefaultFilters
为false
,则Spring
只扫描被明确指定过滤规则的组件,不再包含默认规则(如@Component
、@Service
等)的组件。
在Spring
框架中,我们不仅可以通过@ComponentScan
注解的includeFilters
属性设置包含特定注解的类,还可以通过excludeFilters
属性来排除带有特定注解的类。这个功能对于我们自定义模块的加载非常有用,我们可以通过这种方式,精确控制哪些组件被加载到Spring
的IOC
容器中。下面我们将通过一个具体的示例来说明如何使用@ComponentScan
的excludeFilters
属性来排除带有特定注解的类。
全部代码如下:
首先,创建一个注解Exclude
:
package com.nowjava.demo.annotation;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
public @interface Exclude {
}
定义三个类Elephant
、Monkey
、Tiger
package com.nowjava.demo.bean;
import com.nowjava.demo.annotation.Exclude;
import org.springframework.stereotype.Component;
@Component
@Exclude
public class Elephant {
}
package com.nowjava.demo.bean;
import org.springframework.stereotype.Component;
@Component
public class Monkey {
}
package com.nowjava.demo.bean;
import org.springframework.stereotype.Component;
@Component
public class Tiger {
}
注意,这几个类都标记为 @Component
,Elephant
类上有@Exclude
注解。
接下来,我们创建配置类 FilterConfiguration
,在其中使用 @ComponentScan
注解,并通过 excludeFilters
属性排除所有标记为 @Exclude
的类:
package com.nowjava.demo.configuration;
import com.nowjava.demo.annotation.Exclude;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.FilterType;
@Configuration
@ComponentScan(basePackages = "com.nowjava.demo",
excludeFilters = @ComponentScan.Filter(type = FilterType.ANNOTATION, classes = Exclude.class))
public class FilterConfiguration {
}
这样,在Spring IOC
容器中,只有 Tiger
和 Monkey
类会被扫描并实例化,因为 Elephant
类被 @Exclude
注解标记,而我们在 FilterConfiguration
类中排除了所有被 @Exclude
注解标记的类。
主程序为:
package com.nowjava.demo;
import com.nowjava.demo.configuration.FilterConfiguration;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
public class DemoApplication {
public static void main(String[] args) {
ApplicationContext ctx = new AnnotationConfigApplicationContext(FilterConfiguration.class);
String[] beanNames = ctx.getBeanDefinitionNames();
for (String beanName : beanNames) {
System.out.println(beanName);
}
}
}
运行结果:
总结:这小节主要讲解了如何在Spring
框架中通过@ComponentScan
注解的excludeFilters
属性进行注解过滤,以精确控制加载到Spring IOC
容器中的组件。在本小节的示例中,我们首先创建了一个名为Exclude
的注解,然后定义了三个类Elephant
、Monkey
、和Tiger
,它们都被标记为@Component
,其中Elephant
类上还有一个@Exclude
注解。接下来,我们创建了一个配置类FilterConfiguration
,其中使用了@ComponentScan
注解,并通过excludeFilters
属性排除了所有标记为@Exclude
的类。因此,当程序运行时,Spring IOC
容器中只有Tiger
和Monkey
类会被扫描并实例化,因为Elephant
类被@Exclude
注解标记,所以被排除了。
在Spring
框架中,除了可以通过指定注解来进行包含和排除类的加载,我们还可以利用正则表达式来对组件进行过滤。这种方式提供了一种更灵活的方式来选择需要被Spring IOC
容器管理的类。具体来说,可以利用正则表达式来包含或者排除名称符合某个特定模式的类。下面,我们将通过一个具体的例子来展示如何使用正则表达式过滤来只包含类名以特定字符串结尾的类。
下面的例子将演示如何只包含类名以Tiger
结尾的类。
全部代码如下:
定义三个类Tiger
、Elephant
、Monkey
package com.nowjava.demo.bean;
import org.springframework.stereotype.Component;
@Component
public class Elephant {
}
package com.nowjava.demo.bean;
import org.springframework.stereotype.Component;
@Component
public class Monkey {
}
package com.nowjava.demo.bean;
import org.springframework.stereotype.Component;
@Component
public class Tiger {
}
接着我们创建配置类 FilterConfiguration
,使用 @ComponentScan
注解,并通过 includeFilters
属性来包含类名以 Tiger
结尾的类:
package com.nowjava.demo.configuration;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.FilterType;
@Configuration
@ComponentScan(basePackages = "com.nowjava.demo",
useDefaultFilters = false,
includeFilters = @ComponentScan.Filter(type = FilterType.REGEX, pattern = ".*Tiger"))
public class FilterConfiguration {
}
在上述示例中,我们使用 FilterType.REGEX
过滤类型,并指定要包含的正则表达式模式 ".*Tiger"
。结果只会有 Tiger
类会被Spring的IOC
容器扫描并实例化,因为只有 Tiger
类的类名满足正则表达式 ".*Tiger"
。这里.
代表任意单个字符(除了换行符),*
代表前一个字符重复任意次,.*
组合起来表示匹配任意个任意字符。
主程序:
package com.nowjava.demo;
import com.nowjava.demo.configuration.FilterConfiguration;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
public class DemoApplication {
public static void main(String[] args) {
ApplicationContext ctx = new AnnotationConfigApplicationContext(FilterConfiguration.class);
String[] beanNames = ctx.getBeanDefinitionNames();
for (String beanName : beanNames) {
System.out.println(beanName);
}
}
}
运行结果
总结:本小节介绍了如何在Spring
框架中使用正则表达式对组件进行过滤,以选择哪些类应被Spring IOC
容器管理。在所给示例中,首先定义了三个类Elephant
、Monkey
和Tiger
。然后创建了一个配置类FilterConfiguration
,使用了@ComponentScan
注解,并通过includeFilters
属性设置了一个正则表达式" .*Tiger"
,用于选择类名以"Tiger"
结尾的类。所以在运行主程序时,Spring
的IOC
容器只会扫描并实例化Tiger
类,因为只有Tiger
类的类名满足正则表达式" .*Tiger"
。
"Assignable
类型过滤"是Spring
框架在进行组件扫描时的一种过滤策略,该策略允许我们指定一个或多个类或接口,然后Spring
会包含或排除所有可以赋值给这些类或接口的类。如果我们指定了一个特定的接口,那么所有实现了这个接口的类都会被包含(或者排除)。同样,如果指定了一个具体的类,那么所有继承自这个类的类也会被包含(或者排除)。
在下面的例子中,我们将使用 “Assignable
类型过滤” 来包含所有实现了 Animal
接口的类。
全部代码如下:
首先,我们定义一个 Animal
接口
package com.nowjava.demo.bean;
public interface Animal {
}
接着定义三个类:Elephant
、 Monkey
和Tiger
,其中Tiger
没有实现Animal
接口
package com.nowjava.demo.bean;
import org.springframework.stereotype.Component;
@Component
public class Elephant implements Animal {
}
package com.nowjava.demo.bean;
import org.springframework.stereotype.Component;
@Component
public class Monkey implements Animal {
}
package com.nowjava.demo.bean
import org.springframework.stereotype.Component;
@Component
public class Tiger {
}
然后,我们创建一个 FilterConfiguration
类并使用 @ComponentScan
来扫描所有实现了 Animal
接口的类。
package com.nowjava.demo.configuration;
import com.nowjava.demo.bean.Animal;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.FilterType;
@Configuration
@ComponentScan(basePackages = "com.nowjava.demo",
useDefaultFilters = false,
includeFilters = @ComponentScan.Filter(type = FilterType.ASSIGNABLE_TYPE, classes = Animal.class))
public class FilterConfiguration {
}
这种过滤方式在@ComponentScan
注解中通过FilterType.ASSIGNABLE_TYPE
来使用,这里Spring
将只扫描并管理所有实现了Animal
接口的类。
最后,我们创建一个主程序来测试:
package com.nowjava.demo;
import com.nowjava.demo.configuration.FilterConfiguration;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
public class DemoApplication {
public static void main(String[] args) {
ApplicationContext ctx = new AnnotationConfigApplicationContext(FilterConfiguration.class);
String[] beanNames = ctx.getBeanDefinitionNames();
for (String beanName : beanNames) {
System.out.println(beanName);
}
}
}
运行结果:
这里也可以看到,只有实现了 Animal
接口的类才会被 Spring
的 IoC
容器扫描并实例化,其他的 @Component
类没有实现 Animal
接口的bean
将不会被扫描和实例化。
总结:本小节介绍了Spring
框架中的"Assignable
类型过滤",这是一种可以指定一个或多个类或接口进行组件扫描的过滤策略。Spring
会包含或排除所有可以赋值给这些类或接口的类。在本小节的例子中,首先定义了一个Animal
接口,然后定义了三个类Elephant
、Monkey
和Tiger
,其中Elephant
和Monkey
实现了Animal
接口,而Tiger
没有。然后创建了一个FilterConfiguration
类,使用了@ComponentScan
注解,并通过includeFilters
属性和FilterType.ASSIGNABLE_TYPE
类型来指定扫描所有实现了Animal
接口的类。因此,当运行主程序时,Spring
的IOC
容器只会扫描并实例化实现了Animal
接口的Elephant
和Monkey
类,未实现Animal
接口的Tiger
类不会被扫描和实例化。
Spring
也允许我们定义自己的过滤器来决定哪些组件将被Spring IoC
容器扫描。为此,我们需要实现TypeFilter
接口,并重写match()
方法。在match()
方法中,我们可以自定义选择哪些组件需要被包含或者排除。
全部代码如下:
新增一个接口Animal
package com.nowjava.demo.bean;
public interface Animal {
String getType();
}
定义几个类,实现Animal
接口
package com.nowjava.demo.bean;
import org.springframework.stereotype.Component;
@Component
public class Elephant implements Animal {
public String getType() {
return "This is a Elephant.";
}
}
package com.nowjava.demo.bean;
import org.springframework.stereotype.Component;
@Component
public class Monkey implements Animal {
public String getType() {
return "This is an Monkey.";
}
}
package com.nowjava.demo.bean;
import org.springframework.stereotype.Component;
@Component
public class Tiger implements Animal {
public String getType() {
return "This is a Tiger.";
}
}
package com.nowjava.demo.bean;
import org.springframework.stereotype.Component;
@Component
public class Tiger2 {
public String getType() {
return "This is a Tiger2.";
}
}
Tiger2
没实现Animal
接口,后面用来对比。
下面我们先个自定义一个过滤器CustomFilter
,它实现了TypeFilter
接口,这个过滤器会包含所有实现了Animal
接口并且类名以"T"
开头的类:
package com.nowjava.demo.filter;
import com.nowjava.demo.bean.Animal;
import org.springframework.core.type.ClassMetadata;
import org.springframework.core.type.classreading.MetadataReader;
import org.springframework.core.type.classreading.MetadataReaderFactory;
import org.springframework.core.type.filter.TypeFilter;
import java.io.IOException;
import java.util.Arrays;
public class CustomFilter implements TypeFilter {
@Override
public boolean match(MetadataReader metadataReader, MetadataReaderFactory metadataReaderFactory) throws IOException {
ClassMetadata classMetadata = metadataReader.getClassMetadata();
// 如果全限定类名以 "T" 开头并且实现了 "Animal" 接口
return classMetadata.getClassName().startsWith("com.nowjava.demo.bean.T") &&
Arrays.asList(classMetadata.getInterfaceNames()).contains(Animal.class.getName());
}
}
如果 match
方法返回 true
,那么 Spring
将把这个类视为候选组件,还需满足其他条件才能创建bean
,如果这个类没有使用@Component
、@Service
等注解,那么即使过滤器找到了这个类,Spring
也不会将其注册为bean
。因为Spring
依然需要识别类的元数据(如:@Component
、@Service
等注解)来确定如何创建和管理bean
。反之,如果 match
方法返回 false
,那么 Spring
将忽略这个类。
在match
方法中
metadataReader.getClassMetadata()
返回一个 ClassMetadata
对象,它包含了关于当前类的一些元数据信息,例如类名、是否是一个接口、父类名等。classMetadata.getClassName()
返回当前类的全限定类名,也就是包括了包名的类名。在 match
方法中,我们检查了当前类的全限定名是否以 "com.nowjava.demo.bean.T"
开头,并且当前类是否实现了 "Animal"
接口。如果满足这两个条件,match
方法就返回 true
,Spring
会将这个类视为候选组件。如果这两个条件有任何一个不满足,match
方法就返回 false
,Spring
就会忽略这个类,不会将其视为候选组件。
然后,在我们的FilterConfiguration
中,使用FilterType.CUSTOM
类型,并且指定我们刚才创建的CustomFilter
类:
package com.nowjava.demo.configuration;
import com.nowjava.demo.filter.CustomFilter;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.FilterType;
@Configuration
@ComponentScan(basePackages = "com.nowjava.demo",
useDefaultFilters = false,
includeFilters = @ComponentScan.Filter(type = FilterType.CUSTOM, classes = CustomFilter.class))
public class FilterConfiguration {
}
这样,当Spring IoC
容器进行扫描的时候,只有类名以"T"
开头并且实现了Animal
接口的组件才会被包含。在我们的例子中,只有Tiger
类会被包含,Tiger2
、Elephant
和Monkey
类将被排除。
主程序:
package com.nowjava.demo;
import com.nowjava.demo.configuration.FilterConfiguration;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
public class DemoApplication {
public static void main(String[] args) {
ApplicationContext ctx = new AnnotationConfigApplicationContext(FilterConfiguration.class);
String[] beanNames = ctx.getBeanDefinitionNames();
for (String beanName : beanNames) {
System.out.println(beanName);
}
}
}
运行结果:
调试会发现,match
方法在不停的回调。其实match
方法的调用次数和 Spring
应用上下文中的 Bean
定义数量是相关的,当我们使用 @ComponentScan
进行包扫描时,Spring
会遍历指定包(及其子包)下的所有类,对每个类进行分析以决定是否需要创建对应的 Bean
。
当我们使用 @ComponentScan.Filter
定义自定义的过滤器时,Spring
会为每个遍历到的类调用过滤器的 match
方法,以决定是否需要忽略这个类。因此,match
方法被调用的次数等于 Spring
扫描到的类的数量,不仅包括最终被创建为 Bean
的类,也包括被过滤器忽略的类。
这个行为可能受到一些其他配置的影响。例如,如果 Spring
配置中使用了懒加载 (@Lazy
),那么 match
方法的调用可能会被延迟到 Bean
首次被请求时。
总结:本小节介绍了如何在Spring
框架中创建和使用自定义过滤器,以决定哪些组件将被Spring IoC
容器视为候选组件。通过实现TypeFilter
接口并重写其match()
方法,可以根据自定义的条件决定哪些类会被包含在候选组件的列表中。在这个例子中,我们创建了一个自定义过滤器,只有以"T"
开头且实现了Animal
接口的类才会被标记为候选组件。当Spring
进行包扫描时,会遍历所有的类,并对每个类调用过滤器的match()
方法,这个方法的调用次数等于Spring
扫描到的类的数量。然后,只有那些同时满足过滤器条件并且被Spring
识别为组件的类(例如,使用了@Component
或@Service
等注解),才会被实例化为Bean
并被Spring IoC
容器管理。如果配置了懒加载,那么Bean
的实例化可能会被延迟到Bean
首次被请求时。
Spring
的组件扫描机制提供了一些强大的特性,我们来逐一讲解。
Spring
提供了 @ComponentScans
注解,让我们能够组合多个 @ComponentScan
使用,这样可以让我们在一次操作中完成多次包扫描。
@ComponentScans
的主要使用场景是当需要对Spring
的组件扫描行为进行更精细的控制时,可以在同一个应用程序中扫描两个完全独立的包,也可以在应用多个独立的过滤器来排除或包含特定的组件。
可以看到@ComponentScans
注解接收了一个ComponentScan
数组,也就是一次性组合了一堆 @ComponentScan
注解。
让我们通过一个例子来看看如何使用 @ComponentScans
来组合多个 @ComponentScan
。
全部代码如下:
首先,我们定义两个简单的类,分别在 com.nowjava.demo.bean1
和 com.nowjava.demo.bean2
包中:
package com.nowjava.demo.bean1;
import org.springframework.stereotype.Component;
@Component
public class BeanA {
}
package com.nowjava.demo.bean2;
import org.springframework.stereotype.Component;
@Component
public class BeanB {
}
然后,我们在配置类中使用 @ComponentScans
来一次性扫描这两个包:
package com.nowjava.demo.configuration;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.ComponentScans;
import org.springframework.context.annotation.Configuration;
@Configuration
@ComponentScans({
@ComponentScan("com.nowjava.demo.bean1"),
@ComponentScan("com.nowjava.demo.bean2")
})
public class AppConfig {
}
最后,我们可以测试一下是否成功地扫描到了这两个类:
package com.nowjava.demo;
import com.nowjava.demo.bean1.BeanA;
import com.nowjava.demo.bean2.BeanB;
import com.nowjava.demo.configuration.AppConfig;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
public class DemoApplication {
public static void main(String[] args) {
ApplicationContext ctx = new AnnotationConfigApplicationContext(AppConfig.class);
BeanA beanA = ctx.getBean(BeanA.class);
BeanB beanB = ctx.getBean(BeanB.class);
System.out.println("beanA = " + beanA);
System.out.println("beanB = " + beanB);
}
}
运行上述 main
方法,BeanA
和 BeanB
就成功地被扫描并注入到了 Spring
的 ApplicationContext
中。
运行结果:
总结:本小节介绍了Spring
包扫描机制的一个重要特性,即能够使用@ComponentScans
注解进行组合包扫描。这个特性允许在一次操作中完成多次包扫描,实现对Spring
组件扫描行为的精细控制。例如,可以同时扫描两个完全独立的包,或者应用多个独立的过滤器来排除或包含特定的组件。在本小节的示例中,使用@ComponentScans
一次性扫描了com.nowjava.demo.bean1
和com.nowjava.demo.bean2
两个包,成功地将BeanA
和BeanB
扫描并注入到Spring
的ApplicationContext
中。
当我们在Spring
中使用注解进行bean
的定义和管理时,通常会用到@Component
, @Service
, @Repository
, @Controller
等注解。在使用这些注解进行bean
定义的时候,如果我们没有明确指定bean
的名字,那么Spring
会根据一定的规则为我们的bean
生成一个默认的名字。
这个默认的名字一般是类名的首字母小写。例如,对于一个类名为MyService
的类,如果我们像这样使用@Service
注解:
@Service
public class MyService {
}
那么Spring
会为我们的bean
生成一个默认的名字myService
。我们可以在应用的其他地方通过这个名字来引用这个bean
。例如,我们可以在其他的bean
中通过@Autowired
注解和这个名字来注入这个bean
:
@Autowired
private MyService myService;
这个默认的名字是通过BeanNameGenerator
接口的实现类AnnotationBeanNameGenerator
来生成的。AnnotationBeanNameGenerator
会检查我们的类是否有明确的指定了bean
的名字,如果没有,那么它就会按照类名首字母小写的规则来生成一个默认的名字。
为了解释这个过程,让我们看一下 AnnotationBeanNameGenerator
类的源码,以下源码对应的Spring
版本是5.3.7
。
先给出源码图片,后面给出源码分析
代码块提出来分析:
public String generateBeanName(BeanDefinition definition, BeanDefinitionRegistry registry) {
if (definition instanceof AnnotatedBeanDefinition) { // 该行检查BeanDefinition是否为AnnotatedBeanDefinition
String beanName = this.determineBeanNameFromAnnotation((AnnotatedBeanDefinition)definition); // 该行调用方法来从注解获取bean名称
if (StringUtils.hasText(beanName)) { // 检查是否获取到了有效的bean名称
return beanName; // 如果有,返回这个名称
}
}
return this.buildDefaultBeanName(definition, registry); // 如果没有从注解中获取到有效的名称,调用方法生成默认的bean名称
}
再看看determineBeanNameFromAnnotation
方法
这段代码很长,我们直接将代码块提出来分析:
本文系作者在时代Java发表,未经许可,不得转载。
如有侵权,请联系nowjava@qq.com删除。