Spring注解开发@Bean和@ComponentScan使用案例

组件注册

用@Bean来注册

搭建好maven web工程

pom加入spring-context,spring-core等核心依赖

创建实例类com.hjj.bean.Person, 生成getter,setter方法

public class Person {
  private String name;
  private int age;
}

创建com.hjj.config.MainConfig

@Configuration  //告诉spring是一个配置类
public class MainConfig {

  // 给容器中注册一个Bean,类行为返回值的类型,id默认是用方法名作为id
  @Bean("mikePerson")
  public Person person(){
    return new Person("mike",20);
  }
}

主测试类

public class MainTest {
  public static void main(String[] args) {
    AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(MainConfig.class);
    Person bean = applicationContext.getBean(Person.class);
    System.out.println(bean);

    String[] beanNamesForType = applicationContext.getBeanNamesForType(Person.class);
    for (String type : beanNamesForType) {
      System.out.println(type); //配置类中的方法名,注意:通过修改配置类的@bean value也可以修改
    }
  }
}

@ComponentScan包扫描

配置类中MainConfig.java

@Configuration  //告诉spring是一个配置类
@ComponentScan("com.hjj") // 扫描包的路径
public class MainConfig {

  // 给容器中注册一个Bean,类行为返回值的类型,id默认是用方法名作为id
  @Bean("mikePerson")
  public Person person(){
    return new Person("mike",20);
  }
}

新建测试的com.hjj.controller,service,dao

@Controller
public class BookController {
}

@Repository
public class BookDao {
}

@Service
public class BookService {
}

单元测试

  @Test
  public void test01(){
    AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(MainConfig.class);
    String[] beanDefinitionNames = applicationContext.getBeanDefinitionNames(); //获取所有组件
    for (String beanDefinitionName : beanDefinitionNames) {
      System.out.println(beanDefinitionName);
    }
  }

ComponentScan字段,有includeFilter(),和excludeFilter() 只包含或排除某些组件

@ComponentScan(value = "com.hjj",excludeFilters={@Filter(type=FilterType.ANNOTATION,classes={Controller.class,Service.class})})
@ComponentScan(value = "com.hjj",includeFilters={@Filter(type=FilterType.ANNOTATION,classes={Controller.class,Service.class})},userDefaultFilters=false)

// excludeFilter源码
ComponentScan.Filter[] excludeFilters() default {};

// ComponentScan.Filter源码
@Retention(RetentionPolicy.RUNTIME)
@Target({})
public @interface Filter {
 FilterType type() default FilterType.ANNOTATION;

 @AliasFor("classes")
 Class<?>[] value() default {};

 @AliasFor("value")
 Class<?>[] classes() default {};

 String[] pattern() default {};
}

@ComponentScan被@Repeatable(ComponentScans.class),可以重复写,用来写不同的执行策略。

@ComponentScans 里面可以放ComponentScan类型的值

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE})
@Documented
public @interface ComponentScans {
  ComponentScan[] value();
}

FilterType

public enum FilterType {
  ANNOTATION,  // 按注解扫描
  ASSIGNABLE_TYPE, // 按给定的类型
  ASPECTJ, // 可以用aspectJ表达式
  REGEX, // 正则表达式
  CUSTOM; // 自定义规则

  private FilterType() {
  }
}

ASSIGNABLE_TYPE

@ComponentScan(value = "com.hjj",
includeFilters ={@ComponentScan.Filter(type=FilterType.ASSIGNABLE_TYPE,classes = {BookService.class})})

// config配置如上注解后 bookservice,可以被发现
  @Test
  public void demoTest1(){
    AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(MainConfig.class);
    String[] beanDefinitionNames = applicationContext.getBeanDefinitionNames();
    for (String beanDefinitionName : beanDefinitionNames) {
      System.out.println(beanDefinitionName);
    }
  }

Custom 可以自定义配置或写业务扫描类的信息,match返回true则是加到组件

1.复写TypeFilter的match方法

public class MyTypeFilter implements TypeFilter {

  /**
   *
   * @param metadataReader 读取到的当前正在扫描的类的信息
   * @param metadataReaderFactory 可以获取到其他任何类信息
   * @return
   * @throws IOException
   */
  @Override
  public boolean match(MetadataReader metadataReader, MetadataReaderFactory metadataReaderFactory) throws IOException {
    // 获取当前类注解的信息
    AnnotationMetadata annotationMetadata = metadataReader.getAnnotationMetadata();

    // 获取当前正在扫描类的信息
    ClassMetadata classMetadata = metadataReader.getClassMetadata();

    // 获取当前类资源(类的路径等)
    Resource resource = metadataReader.getResource();

    String className = classMetadata.getClassName();
    System.out.println("className:" + className);
    return false;
  }
}

2. 加上注解

@Configuration  //告诉spring是一个配置类
@ComponentScan(value = "com.hjj",
includeFilters ={
    @ComponentScan.Filter(type=FilterType.CUSTOM,classes = {MyTypeFilter.class})},
    useDefaultFilter=false)
public class MainConfig {

}

@Test
public void test01(){
AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(MainConfig.class);
String[] beanDefinitionNames = applicationContext.getBeanDefinitionNames();
for (String beanDefinitionName : beanDefinitionNames) {
System.out.println(beanDefinitionName);
}
}

结果

className:com.hjj.test.IOCTest
className:com.hjj.MainTest
className:com.hjj.bean.Person
className:com.hjj.config.MyTypeFilter
className:com.hjj.demo.DemoTest
className:com.hjj.demo.Employee
className:com.hjj.demo.Manager
org.springframework.context.annotation.internalConfigurationAnnotationProcessor
org.springframework.context.annotation.internalAutowiredAnnotationProcessor
org.springframework.context.annotation.internalCommonAnnotationProcessor
org.springframework.context.event.internalEventListenerProcessor
org.springframework.context.event.internalEventListenerFactory
mainConfig

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Spring启动后获取所有拥有特定注解的Bean实例代码

    本文研究的主要是Spring启动后获取所有拥有特定注解的Bean,具体如下. 最近项目中遇到一个业务场景,就是在Spring容器启动后获取所有的Bean中实现了一个特定接口的对象,第一个想到的是ApplicationContextAware,在setApplicationContext中去通过ctx获取所有的bean,后来发现好像逻辑不对,这个方法不是在所有bean初始化完成后实现的,后来试了一下看看有没有什么Listener之类的,发现了好东西ApplicationListener,然后百度一

  • Spring注解驱动扩展原理BeanFactoryPostProcessor

    1.扩展原理-BeanFactoryPostProcessor BeanFactoryPostProcessor * 扩展原理: * BeanPostProcessor:bean后置处理器,bean创建对象初始化前后进行拦截工作的 * * 1.BeanFactoryPostProcessor:beanFactory的后置处理器: * 在BeanFactory标准初始化之后调用,来定制和修改BeanFactory的内容: * 所有的bean定义已经保存加载到beanFactory,但是bean的实

  • Java之Spring注解配置bean实例代码解析

    前面几篇均是使用xml配置bean,如果有上百个bean,这是不可想象的.故而,请使用注解配置bean !!! [1]注解类别 @Component : 基本注解, 标识了一个受 Spring(点击这里可以下载<Spring应用开发完全手册>) 管理的组件 @Repository : 标识持久层组件 @Service : 标识服务层(业务层)组件 @Controller : 标识表现层组件 Spring 能够从 classpath 下自动扫描, 侦测和实例化具有特定注解的组件. 对于扫描到的组

  • Spring如何使用注解的方式创建bean

    这篇文章主要介绍了Spring如何使用注解的方式创建bean,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 第一种使用配置类的方式 1.创建一个bean package com.springbean; public class Person { private String name; private Integer age ; public Person(String name, Integer age) { this.name = name

  • springboot @ConditionalOnMissingBean注解的作用详解

    @ConditionalOnMissingBean,它是修饰bean的一个注解,主要实现的是,当你的bean被注册之后,如果而注册相同类型的bean,就不会成功,它会保证你的bean只有一个,即你的实例只有一个,当你注册多个相同的bean时,会出现异常,以此来告诉开发人员. 代码演示 @Component public class AutoConfig { @Bean public AConfig aConfig() { return new AConfig("lind"); } @B

  • Spring @Bean注解配置及使用方法解析

    使用说明 这个注解主要用在方法上,声明当前方法体中包含了最终产生 bean 实例的逻辑,方法的返回值是一个 Bean.这个 bean 会被 Spring 加入到容器中进行管理,默认情况下 bean 的命名就是使用了 bean 注解的方法名.@Bean 一般和 @Component 或者 @Configuration 一起使用. @Bean 显式声明了类与 bean 之间的对应关系,并且允许用户按照实际需要创建和配置 bean 实例. 该注解相当于: <bean id="useService

  • Springboot @Configuration @bean注解作用解析

    这篇文章主要介绍了springboot @Configuration @bean注解作用解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 @Configuration注解可以达到在Spring中使用xml配置文件的作用 @Bean就等同于xml配置文件中的<bean> 在spring项目中我们集成第三方的框架如shiro会在spring.xml配置文件中进行配置,例如: <!-- 配置shiro框架提供过滤器工厂 --> <

  • 简单了解Spring Bean常用注解的装配

    这篇文章主要介绍了简单了解Spring Bean常用注解的装配,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 基于注解的装配 在Spring框架中,尽管使用XML配置文件可以很简单地装配Bean,但如果应用中有大量的Bean需要装配,会导致XML配置文件过于庞大,不方便以后的升级与维护,因此更多的时候推荐开发者使用注解(annotation)的方式去装配Bean. 在Spring框架中定义了一系列的注解,下面介绍集中常用的注解. @Compon

  • Spring Bean管理注解方式代码实例

    1.使用注解的方式需要配置applicationContext.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:context=&qu

  • Spring @Bean vs @Service注解区别

    今天跟同事讨论了一下在Spring Boot中,是使用@Configuration和@Bean的组合来创建Bean还是直接使用 @Service等注解放在类上的方式.笔者倾向于使用第一种,即@Configuration和@Bean的组合. 先来看一个例子,目标是创建SearchService的一个Bean. 直接使用@Service的方式: // SearchService.java package li.koly.search; import java.util.List; public in

随机推荐