springboot加载注入bean的方式

在SpringBoot的大环境下,基本上很少使用之前的xml配置Bean,主要是因为这种方式不好维护而且也不够方便。
springboto注入bean主要采用下图几种方式,分为本地服务工程注解声明的bean和外部依赖包中的bean。
在这里插入图片描述

一、 springboot装配本地服务工程中的bean

1、注解装配Bean

1、使用Component,Service,Controller,Repository等派生注解
只要在类上加类上加 @Component 注解即可,该注解只要被扫描到就会注入到spring的bean容器中。

@Component
public class AnoDemoBean {
}

当然不只是@Component注解可以声明Bean,还有如:@Repository、@Service、@Controller 等常用注解同样可以。

如果去看这些注解,就发现这些注解上本身就有加 @Component 注解

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component   //可以看到@Service注解上有添加@Component,  @Repository和@Controller也一样。        
public @interface Service {
    @AliasFor(
        annotation = Component.class
    )
    String value() default "";
}

这系列注解的出现,给我们带来了极大的便利。我们不需要像以前那样在bean.xml文件中配置bean了,现在只用在类上加上相关注解,就能轻松完成bean的定义。

这四种注解在功能上其实没有特别的区别,不过在业界有个不成文的约定:

  • Controller 一般用在控制层
  • Service 一般用在业务层
  • Repository 一般用在数据层
  • Component 一般用在公共组件上

2、@Bean定义方式

这种方式,主要是结合Configuration来定义bean,首先是声明一个配置类,而后再配置类中,经过返回bean对象的方法形式来声明bean,通常使用姿式以下

@Data
public class ConfigDemoBean {
}

@Configuration
public class BeanLoadConfig {
    @Bean
    public ConfigDemoBean configDemoBean() {
        return new ConfigDemoBean();
    }
}

须要说明的一点是BeanLoadConfig类自己也被Spring容器看为一个Bean。

3、Component注解 VS Bean注解

1)作用对象不同:@Component 注解作用于类,而 @Bean 注解作用于方法。

@Bean 方式更加灵活,比如当我们引用第三方库中的类需要装配到 Spring 容器时,只能通过 @Bean 来实现。

比如

@Configuration
public class WireThirdLibClass {
    @Bean
    public ThirdLibClass getThirdLibClass() {
        //第三方的ThirdLibClass类
        return new ThirdLibClass();
    }
}

再比如

@Bean
public OneService getService(status) {
    case (status)  {
        when 1:
                return new serviceImpl1();
        when 2:
                return new serviceImpl2();
        when 3:
                return new serviceImpl3();
    }
}

这两点都是@Component无法做到,只能@Bean实现,所以说@Bean更加灵活。

2)@Component通常是通过类路径扫描来自动装配到Spring容器中。而@Bean通常我们会在该注解的方法中定义产生这个bean的逻辑。

我们可以加一些@Conditional,@ConditionalOnBean等等一些注解来控制是否声明该Bean,不会一开始就自动装配到Spring容器中。

比如

public class MacCondition implements Condition {

   @Override
    public boolean matches(ConditionContext conditionContext, AnnotatedTypeMetadata annotatedTypeMetadata) {
        Environment environment = conditionContext.getEnvironment();
        String property = environment.getProperty("os.name");
        if (property.contains("Mac")) {
            log.info("当前操作系统是:Mac OS X");
            return true;
        }
        return false;
    }
}

@Configuration
public class ConditionalConfig {
    /**
     * 如果MacCondition的实现方法返回true,则注入这个bean
     */
    @Bean("mac")
    @Conditional({MacCondition.class})
    public SystemBean systemMac() {
        log.info("ConditionalConfig方法注入 mac实体");
        return new SystemBean("Mac ios系统","001");
    }
}

上面的例子表示,如果当前操作系统是Mac,才会注入当前Bean。这个也只能 @Bean 注解才能实现。

总结:@Component和@Bean都是用来注册Bean并装配到Spring容器中,但是Bean比Component的自定义性更强。可以实现一些Component实现不了的自定义加载类。

二、springboot装配依赖包中的bean

当SpingBoot主类启动的时候,@SpringBootApplication注解会默认去扫描的自己所在包路径和它的子包路径下的所有需要装配的类,自动装配到spring的bean容器中。
但是如果你提供了一个Jar包供第三方用户使用,那么你这个jar包中的Bean,能被第三方加载么?
这就要看你当前项目的包名和你引用的第三方Jar包的包名是否一致了?如果一致,第三方依赖包中的bean可以直接加载,如果不一致,则无法加载第三方依赖中的bean。
例如,如果你当前项目本包的地址是com.jincou,而你引用的第三方Jar的本包也是 com.jincou,那么第三方Jar的Bean就可以被扫描到并注入到spring的容器中。如果你当前项目本包的地址是com.jincou ,而你引用的第三方Jar的本包是 com.third,那么也就是第三方Jar的Bean无法被扫描到,所以也就无法注入到Spring容器中。

比如这里有个第三方的Bean。要如何做才能被扫描注入到Spring容器中呢。

package com.third.bean;

import org.springframework.stereotype.Component;

/**
 * @Description: 这个bean作为第三方bean 给依赖该jar包的项目使用
 */
@Component
public class ThirdComponentBean {
    private String type = "第三方ThirdComponent注解生成bean实体";
}

1、ComponentScan注解

很简单,既然@SpringBootApplication注解默认扫描只是当前项目的本包和它的子包,那就想办法让它扫描第三方jar的包就好了。

/**
 * @Description: Springboot 启动类
 */
@ComponentScan(basePackages ={"com.third.bean"})
@SpringBootApplication()
public class Application {

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

@ComponentScan主要就是定义扫描的路径从中找出标识了需要装配的类自动装配到spring的bean容器中。

需要装配的类也就是上面加了@Controller,@Service,@Repository,@Component,@Configuration等等的注解的Bean到IOC容器中。

这里不一定要加在启动类上,你可以加在加在装配的类上,但建议加在启动类上,比较直观,后期如果要改动或者去除也比较好找。

2、Import注解

@ComponentScan是扫描整个包,但其实你可能只需注入一个或者几个指定的Bean,那我们可以考虑用 @Import 注解

@Import(value= com.third.bean.ThirdComponentBean.class)
@SpringBootApplication()
public class Application {

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

这样做同样也会成功的将ThirdComponentBean对象注入到Spring的bean容器中。

3、spring.factories文件中配置bean

上面两种注入方式都有个很明显缺点,就是如果我需要引用外部jar包的Bean的时候,都需要在当前项目配置 @ComponentScan 或者 @Import 去扫描才能注入当前Bean,这样显然不够友好。

可不可以当前项目什么都不做就可以直接引用第三方jar的Bean呢?

当然可以。

我们只需要在将配置放在第三方jar指定的文件中即可,使用者会自动加载,从而避免的代码的侵入

在资源目录下新建目录 META-INF
在 META-INF 目录下新建文件 spring.factories
在文件中添加下面配置

org.springframework.boot.autoconfigure.EnableAutoConfiguration=com.third.bean.ConfigurationBean
### 回答1: Spring Boot支持动态注入Bean,可以通过编程方式向应用程序上下文中添加Bean。以下是实现动态注入Bean的步骤: 1. 创建一个实现了BeanFactoryPostProcessor接口的类,该类可以在应用程序上下文加载之前修改Bean定义。 2. 在BeanFactoryPostProcessor实现类中,使用BeanDefinitionRegistry接口向应用程序上下文中添加Bean定义。 3. 在添加Bean定义时,可以使用GenericBeanDefinition类创建一个新的Bean定义,并设置Bean的属性和依赖项。 4. 在应用程序上下文加载之后,新添加的Bean将可用于注入。 需要注意的是,动态注入Bean可能会影响应用程序的性能和可维护性,因此应该谨慎使用。 ### 回答2: springboot是一个非常方便的Java开发框架,它提供了很多工具来简化开发过程。其中一个非常重要的功能就是动态注入bean。这个功能允许我们在运行时动态地向Spring容器中添加或删除bean,这样我们就可以根据需要来动态地修改应用程序的行为。 要实现动态注入bean,我们首先需要定义一个标记接口,例如: ``` public interface MyBeanMarker { } ``` 然后我们需要创建一个类来实现这个接口,并且需要标记为@Component: ``` @Component public class MyBean implements MyBeanMarker { // ... } ``` 接下来,我们需要创建一个容器类来管理我们的bean: ``` @Component public class MyBeanContainer { private Map<String, MyBeanMarker> beans = new HashMap<>(); @Autowired public MyBeanContainer(List<MyBeanMarker> beans) { for (MyBeanMarker bean : beans) { this.beans.put(bean.getClass().getName(), bean); } } public MyBeanMarker getBean(String name) { return beans.get(name); } public void addBean(MyBeanMarker bean) { beans.put(bean.getClass().getName(), bean); } public void removeBean(MyBeanMarker bean) { beans.remove(bean.getClass().getName()); } } ``` 在这个容器类中,我们使用@Autowired注解来注入所有实现了MyBeanMarker接口的bean,并且将它们存储在一个Map中。然后我们可以使用容器类的getBean方法来获取bean,使用addBean方法来添加bean,使用removeBean方法来删除bean。 最后,我们需要在我们的应用程序中使用这个容器类来管理我们的bean: ``` @SpringBootApplication public class Application implements CommandLineRunner { @Autowired private MyBeanContainer beanContainer; public static void main(String[] args) { SpringApplication.run(Application.class, args); } @Override public void run(String... args) throws Exception { MyBeanMarker bean = beanContainer.getBean("com.example.MyBean"); if (bean != null) { // 使用bean } // 添加bean MyBean newBean = new MyBean(); beanContainer.addBean(newBean); // 删除bean beanContainer.removeBean(newBean); } } ``` 在这个例子中,我们使用@Autowired注解来注入MyBeanContainer,并且在run方法中使用它来获取、添加和删除bean。当我们需要动态修改应用程序时,我们可以添加或删除bean,并且根据需要使用它们。 总之,动态注入bean是一个非常有用的功能,它使我们能够根据需要动态地修改应用程序的行为。在Spring Boot中,实现动态注入bean非常容易,只需要使用一个简单的容器类来管理所有的bean即可。 ### 回答3: SpringBoot是基于Spring框架的应用程序开发框架,它以简洁的配置和快速的开发为出发点,提供了一套全面的解决方案,帮助我们构建鲁棒性强、易于维护、高效的应用程序。而在SpringBoot中,动态注入Bean就是其中一个非常重要的功能。 动态注入Bean是指在运行时根据特定的条件,动态地向Spring容器中注入Bean对象,这个过程实际上是通过通过实现SpringBeanDefination和BeanFactory来实现的。可以通过编写代码控制Bean加载的条件和实现逻辑,达到在应用程序运行期间动态实现Bean加载的目的,这样带来的好处是提升了应用程序的灵活性和扩展性。 在SpringBoot中,动态注入Bean可以通过以下几种方式实现: 1. 通过@Component注解实现动态注入。通过在类上加上@Component、@Service等注解,SpringBoot会将其作为Bean自动加载,可以通过在运行过程中动态改变其属性或增加其子类来动态注入Bean。 2. 通过实现BeanFactory接口实现动态注入。在SpringBoot中,BeanFactory是所有Bean加载的工厂类,通过实现BeanFactory接口,可以根据运行时的条件来动态生成Bean对象,实现Bean动态注入的功能。 3. 通过自定义注解实现动态注入。指定需要注入Bean的类型和其它参数,运行时扫描带有该注解的类,然后根据注解参数来实现Bean动态注入的功能。 总之,动态注入Bean使得SpringBoot应用程序能够在运行过程中根据需要动态加载Bean对象,提供了更加灵活和扩展的方式来实现应用程序的业务逻辑。掌握动态注入Bean的相关知识,对于开发高质量的SpringBoot应用程序是非常必要的。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值