今天面了一个人,问了个JavaConfig(每天一题,一起进大厂)

写在前面

今天面试了几个人,我都有意问到javaConfig这个概念,熟悉的朋友应该知道这个隶属于SpringBoot体系里。但是最后几个人回答的都不尽如人意。趁此,我们一起来看下这个东西怎么理解。

一般同学听到这个肯定会懵逼掉,不太理解这个词的究竟面试官想要考什么,其实目的就是考察Spring的注解,如何注册成bean,也就是从SpringBoot作为切入点回答。

提前备注下:

传统spring一般都是基于xml配置的,不过后来新增了许多JavaConfig的注解。特别是springboot,基本都是清一色的java config

注解

耳熟能详的第一个可能就是@RestController

spring4为了更方便的支持restfull应用的开发,新增了RestController的注解,比Controller注解多的功能就是给底下的RequestMapping方法默认都加上ResponseBody注解,省得自己再去每个去添加该注解。

这就是java Config的概念所应用。

接着再说@Configuration

使用@Configuration 声明配置类时有两种方法来生成bean

以上两种方法可能在SpringBoot开发当中应用的比较多,我们看个例子

第一种方法:


// 这个会交由Spring容器托管,注册到容器中,因为他本来就是一个@Component
// @Configuration 代表这是一个配置类,相当于applicationContext.xml
@Configuration
public class TestConfig {

    //注册一个bean,就相当于 bean标签
    //方法名,相当于bean标签中的id属性
    //方法返回值,相当于bean标签的class属性
    @Bean
    public User getUser() {
        return new User();//返回要注入的bean对象
    }

}

第二种

// 这个会由Spring容器托管,注册到容器中,因为他本来就是一个@Component
// @Configuration 代表这是一个配置类,相当于applicationContext.xml
@Configuration
@Component("com.pojo")
//将其他配置文件融合到当前的类中
@Import(com.config.TestConfig.class)
public class TestConfig {
    public User getUser() {
        return new User();
    }

}
public class User {
    private String name;

    public String getName() {
        return name;
    }
    @Value("苏世")
    public void setName(String name) {
        this.name = name;
    }
    @Override
    public String toString() {
        return "User{" +
                "name='" + name + ''' +
                '}';
    }
}
public class Test {
    public static void main(String[] args) {
        //如果完全使用了配置类方式去做,就只能通过 AnnotationConfig 上下文来获取容器,通过配置类的class对象加载
        ApplicationContext context = new AnnotationConfigApplicationContext(TestConfig.class);
        User user = (User) context.getBean("getUser");
        System.out.println(user.getName());
    }
}

这就是一个最基本的案例,实现起来非常的简单。下面我们着重分析一下注解的作用,为什么能实现类似于Spring中XML文件一样的作用。

分析源码

想要了解为什么@Configuration会有这样的作用,我们可以跟进去这个注解看看。

进去之后我们会发现,这个注解标签是一个元注解,由很多其他的注解实现,有一个我们应该很熟悉,那就是@Component,有着了这个注解就可以被@ComponentScan扫描并处理。Spring5.0已经自动扫描了,不需要我们自己再去添加了。

首先是@AliasFor标签:

在Spring的众多注解中,经常会发现很多注解的不同属性起着相同的作用,比如@RequestMapping的value属性和path属性,这就需要做一些基本的限制,比如value和path的值不能冲突,比如任意设置value或者设置path属性的值,都能够通过另一个属性来获取值等等。为了统一处理这些情况,Spring创建了@AliasFor标签。

然后是value() :

意思是默认的值就是空,此时我们就可以指定@Configuration(value="属性值")的这种方式,因为只有一个value所以value可以省去不写。

最后是proxyBeanMethods:

有了 proxyBeanMethods 属性后,配置类不会被代理了。主要是为了提高性能,如果你的 @Bean 方法之间没有调用关系的话可以把 proxyBeanMethods 设置为 false。否则,方法内部引用的类生产的类和 Spring 容器中类是两个类。

运行角度分析

看到这里,可能就要深入到Spring的源码中看了。Spring容器启动时,ApplicationContext接口的实现类AnnotationConfigApplicationContext会执行refresh方法,往BeanFactory注册bean就在此方法完成。我们看到这个refresh是核心。我们进入到这个源码中看看:

我截取了其中一部分的源码,在里面有一个方法很关键,那就是invokeBeanFactoryPostProcessors,意思是我们Spring容器首先会初始化BeanFactory,然后激活各种beanFactory处理器,也就是执行invokeBeanFactoryPostProcessors,我们看看这个方法:

在这个方法的内部的核心是ConfigurationClassPostProcessor,这个方法看到@Configuration,就会开启类的加载,这里也就是bean的加载。剩下的越挖越深,源码也越来越深。大体步骤我们可以总结一下:

总结

ConfigurationClassPostProcessor处理器解析@configuration配置类主要过程:

(1)Spring容器初始化时注册ConfigurationClassPostProcessor

(2)Spring容器初始化执行refresh()方法中调用ConfigurationClassPostProcessor

(3)ConfigurationClassPostProcessor处理器借助ConfigurationClassParser完成配置类解析

(4)ConfigurationClassParser配置内解析过程中完成嵌套的MemberClass、@PropertySource注解、@ComponentScan注解(扫描package下的所有Class并进行迭代解析,主要是@Component组件解析及注册)、@ImportResource、@Bean等处理

(5)接下来完成@Bean注册, @ImportResource指定bean的注册以及@Import的bean注册

(6)有@Bean注解的方法在解析的时候作为ConfigurationClass的一个属性,最后还是会转换成BeanDefinition进行处理, 而实例化的时候会作为一个工厂方法进行Bean的创建

现在大致应该明白了,其实一句话说完,还是想办法识别注解,完成和XML一样的功能。

弦外之音

其实这个知识点,大家在学习SpringBoot的自动配置注解那块是一样的。

展开阅读全文

更新时间:2024-09-09

标签:几个人   注解   初始化   容器   处理器   源码   属性   作用   标签   方法

1 2 3 4 5

上滑加载更多 ↓
推荐阅读:
友情链接:
更多:

本站资料均由网友自行发布提供,仅用于学习交流。如有版权问题,请与我联系,QQ:4156828  

© CopyRight 2008-2024 All Rights Reserved. Powered By bs178.com 闽ICP备11008920号-3
闽公网安备35020302034844号

Top