Spring中Bean的三种注入方式

用法上可以分为三种,但是底层实现代码都是统一BeanFactory,这三种也有联系xml注入和annotation注入都是依赖BeanDefinition扩展的接口,注解也是从xml过渡过来的

XML注入

配置过程比较繁琐,但是对代码的侵入性较小,配置和代码分离操作。

实体定义

定义两个属性id,name,并实现get/set方法,重写toString方法,方便看打印结果。

public class UserXml {
    private String id;
    private String name;

    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return "User{" +
                "id='" + id + '\'' +
                ", name='" + name + '\'' +
                '}';
    }
}

xml定义

在resources下新建目录META-INF下建spring-bean.xml文件,并填充对应的bean配置,bean需要配置id或者name值,IOC容器唯一即可,class配置定义的实体路径,对应的property设置初始化属性。

<?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
        https://www.springframework.org/schema/beans/spring-beans.xsd">
    <bean id="user" class="cn.cnzcb.spring.bean.UserXml">
        <property name="id" value="11"/>
        <property name="name" value="java圈"/>
    </bean>
</beans>

输出

创建一个BeanFactory对象,用ClassPathXmlApplicationContext实例化,简单说一下BeanFactory作为IOC容器的底层基础,可以说IOC容器就是BeanFactory,ClassPathXmlApplicationContext是IOC容器的功能扩展;ClassPathXmlApplicationContext需要传入资源文件的路径,在通过getBean方法获取具体的实体类,就是结果输出。

        //xml注入
        BeanFactory classPathXmlApplicationContext = new ClassPathXmlApplicationContext("classpath:/META-INF/spring-bean.xml");
        UserXml userXml = classPathXmlApplicationContext.getBean(UserXml.class);
        System.out.println("userXml XML注入对象:" + userXml);

注解注入

实体定义

public class UserAnnotation {
    private String id;
    private String name;

    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return "User{" +
                "id='" + id + '\'' +
                ", name='" + name + '\'' +
                '}';
    }

注解配置

正常情况下我们是通过标准注解@Configuration进行扫描注入,我们这里直接配置类即可,在这个类里面实例化bean组件,并进行初始化操作。

@Configuration
public class UserConfiguration {
    @Bean
    public UserAnnotation userAnnotation(){
        UserAnnotation userAnnotation = new UserAnnotation();
        userAnnotation.setId("11");
        userAnnotation.setName("java圈");
        return userAnnotation;
    }
}

输出

AnnotationConfigApplicationContext也是BeanFactory的一种实现,和ClassPathXmlApplicationContext功能类似,只是加载渠道不一样,把定义的配置类注册到IOC容器,调用register方法,这里需要注意,下一步需要调refresh方法来实现bean的装载工作,然后通过getBean获取具体的实体,就行输出。

        //注解注入
        AnnotationConfigApplicationContext  annotationConfigApplicationContext = new AnnotationConfigApplicationContext();
        annotationConfigApplicationContext.register(UserConfiguration.class);
        annotationConfigApplicationContext.refresh();
        UserAnnotation userAnnotation = annotationConfigApplicationContext.getBean(UserAnnotation.class);
        System.out.println("UserAnnotation注解注入"+userAnnotation);

BeanDefinition注入

BeanDefinition是BeanFactory的底层实现,包括上面提到的方式,底层也是基于BeanDefinition实现的,一个bean组件对应一个BeanDefinition,但是实际操作过程中不会这个用,只是仅供参考。

实体定义

public class UserBeanDefinition {
    private String id;
    private String name;

    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return "User{" +
                "id='" + id + '\'' +
                ", name='" + name + '\'' +
                '}';
    }
}

输出

通过BeanDefinitionBuilder的genericBeanDefinition实例化一个构造器,传入的参数就是实体类,构建之后做初始化操作,之后BeanDefinition声明调用getBeanDefinition方法,通过getPropertyValues回去bean的具体参数进行结束输出。

        //BeanDefinition注入
        BeanDefinitionBuilder definitionBuilder = BeanDefinitionBuilder.genericBeanDefinition(UserBeanDefinition.class);
        definitionBuilder.addPropertyValue("id", "11");
        definitionBuilder.addPropertyValue("name", "java圈");
        BeanDefinition beanDefinition = definitionBuilder.getBeanDefinition();
        String beanClassName = beanDefinition.getBeanClassName();
        MutablePropertyValues mutablePropertyValues = beanDefinition.getPropertyValues();
        String id = mutablePropertyValues.getPropertyValue("id").getValue().toString();
        String name = mutablePropertyValues.getPropertyValue("name").getValue().toString();
        System.out.println("BeanDefinition注入对象UserBeanDefition{id=" + id+",name="+name+"}");

结果输出

userXml XML注入对象:User{id='11', name='java圈'}
UserAnnotation注解注入User{id='11', name='java圈'}
BeanDefinition注入对象UserBeanDefition{id=11,name=java圈}

转载:https://zhuanlan.zhihu.com/p/102794042


   转载规则


《Spring中Bean的三种注入方式》 锦泉 采用 知识共享署名 4.0 国际许可协议 进行许可。
 上一篇
Shiro框架搭建 Shiro框架搭建
一键构建系统整合Shiro权限控制框架
2021-06-09
下一篇 
ThreadLocal ThreadLocal
WeakReference弱引用对象相对软引用对象具有更短暂的生命周期,只要 G C 发现它仅有弱引用,不管内存空间是否充足,都会回收它,不过 G C 是一个优先级很低的线程,因此不一定会很快发现那些仅有弱引用的对象。 只有对象仅被 Wea
2021-06-03
  目录