• 面试题-Spring和Springboot框架


    前言

    spring框架部分的题目,是我根据Java Guide的面试突击版本V3.0再整理出来的,其中,我选择了一些比较重要的问题,并重新做出相应回答,并添加了一些比较重要的问题,希望对大家起到一定的帮助。

    系列文章:

    面试题-Java基础

    面试题-Java集合

    面试题-Java多线程基础、实现工具和可见性保证

    面试题-线程池和原子变量

    面试题-Java虚拟机

    面试题-计算机网络1

    面试题-计算机网络-HTTP部分


    Spring框架

    1. 什么是 Spring 框架?

      Spring是一种轻量级开发框架,旨在提高开发人员的开发效率以及系统的可维护性。我们说的Spring框架是很多模块的集合,包括核心容器,数据访问集成,web,AOP等等。

    2. @RestController vs @Controller和区别?

      单独使用Controller时,适用于前后端不分离的应用,Controller返回一个model and view。

      RestController将返回数据写入ResponseBody中,Controller+ResponseBody注解一起用和RestController是一样的功能。

    3. 简单说说你对IOC的理解

      IOC叫做控制反转,控制反转是一种设计思想,简单来说就是,本来把程序员控制的流程交给框架控制,框架提供给程序员一些扩展点,程序员实现扩展点就可以方便的调用框架做需要的事情。

      SpringIoc容器是控制反转设计思想的一种实现,程序员把管理对象生命周期的工作交给框架来做,通过配置文件或注解配置需要的对象,在使用的时候跟Spring容器要就可以了。

    4. 谈谈你对AOP的理解

      AOP是面向切面编程,通过把和业务无关,但是却被很多业务模块调用的系统功能封装起来,减少系统的重复代码,提高代码的复用性并提高的系统的维护性。

      比如实际项目中,中间件组的web层代码AOP框架规范是我制定的,统一处理异常,记录日志等操作是放在这个AOP中的,具体用的是ControllerAdvice实现的。

    5. AOP的实现原理你了解吗?简单说说

      AOP的实现最核心的就是利用JDK的动态代理或者cglib实现,项目启动时通过反射获取相关的信息,在创建代理对象时,如果被代理对象实现了接口,就是用JDK动态代理,如果没有实现接口,就使用cglib创建被代理对象的子类来实现代理。

    6. Spring的AOP和AspectJ的AOP有什么区别?

      二者都是AOP思想的实现,AspectJ的功能更加强大。

      • springAOP是基于运行时动态增强

      • AspectJ是编译时操作字节码增强,运行时更快

      项目中遇到无法使用Spring来管理对象这种情况时,就会考虑使用AspectJ来增强对象

    7. 为什么需要bean的作用域?

      spring容器需要管理对象的生命周期,spring通过scope作用域的概念给使用者对对象生命周期的更精细的控制。

    8. bean的作用域都有哪些?

      • singleton:单例,会一直存活直到容器关闭,spring默认的对象作用域是单例的
      • prototype:多实例,每次请求都会创建一个新的对象,交给使用方以后,spring就不负责管理对象的生命周期了
      • request:每次请求都创建一个实例,请求完毕后销毁对象
      • session:每个session创建一个实例,session失效后销毁对象实例
    9. Spring 中的单例 bean 的线程安全问题了解吗?

      对单例bean的非静态成员变量的读写会有线程安全问题。

      如果每个线程对成员变量的修改业务上没有关联,可以通过使用ThreadLocal来保存每个线程单独的变量拷贝副本来实现线程安全。

    10. @Component 和 @Bean 的区别是什么?

      • 作用对象不同:component作用于类,bean作用于方法。
      • bean注解适用于引用第三方类库的bean时使用,因为没办法使用component注解用于第三方类库的源码
    11. 将⼀个类声明为Spring的 bean 的注解有哪些?

      • Component:通用注解
      • Service:Service层的注解
      • Controller:控制层的注解
      • Repository:DAO层的注解
    12. Spring 中的 bean ⽣命周期?

      • Spring先对bean进行实例化
      • Spring将值和引用注入到相关属性中
      • 如果bean实现了相关的aware接口,spring就会调用set方法把相关需求的实例注入到bean中
      • 如果实现了 beanpostProcessor接口,会调用before方法
      • 如果有postConstruct注解,会调用注解的方法
      • 如果实现了InitialingBean接口,会调用afterPropertiesSet方法
      • 如果指定了init-method,会调用init-method
      • 最后调用beanpostProcessor的after方法
      • 销毁前,会调用DispoableBean的destory方法
      • 如果指定了destroy-method,调用这个方法
      • 最后调用 preConstruct注解修饰的方法

      流程图如下:

    1. 你知道spring MVC的工作原理吗?

      • DispatcherServlet接收用户的请求后,根据请求向处理器映射器查询具体处理器
      • 处理器映射器返回对应的处理器链返回给dispatherServlet
      • dispatcherServlet会将请求发送给具体的处理器,并等待处理
      • 处理器处理完毕后,会返回逻辑视图名和模型给前端控制器
      • 前端控制器根据逻辑视图名向视图解析器解析具体的视图
      • 最后视图负责根据模型数据渲染页面,并返回给客户端
    2. MVC的注解都用过什么?

      • RestController:指定控制器
      • RequestMapping,负责编写具体的url,指定post方法,并指定返回内容类型
      • RequestBody:接收通过body传递过来的JSONObject数据参数
      • ControllerAdvice和ExceptionHandler:负责controller的全局异常处理
    3. Spring 框架中⽤到了哪些设计模式?

      • 单例模式:spring的bean作用域默认是单例的
      • 代理模式:AOP就是代理模式的实现
      • 工厂模式:applicationContext就是对象工厂
      • 观察者模式:spring的事件机制就是观察者模式的一个典型应用
    4. Spring 管理事务的⽅式有⼏种?

      • 编程式事务:硬编码
      • 声明式事务:
        • 基于XML
        • 基于注解
    5. Spring事务的隔离级别都有哪些,如何设置?

      和MySQL中的事务隔离级别一样,有四种隔离级别可以设置,还有一种是使用数据库的默认隔离级别

      在transactional注解中设置isolation属性即可

    6. Spring中事务的传播机制有哪些?

      事务的传播机制指的是,如果事务方法互相调用,事务应该如何传播,具体分为三大类:

      • 支持当前事务的:如果有事务,加入当前事务
      • 不支持当前事务的:如果有事务,挂起或者抛出异常
      • 其他情况:嵌套事务,父事务回滚,子事务也会回滚

    我们的具体业务中,相关联的业务都写在了同一个service方法中,所以不存在事务的传播机制问题

    如果需要拆分方法,那么需要根据具体业务逻辑具体分析使用哪种事务

    事务的传播机制

    1. @Transactional(rollbackFor = Exception.class)注解了解吗?

      了解,如果不这么配置,那么事务只会在运行时异常的时候才会回滚,我们也需要cover到非运行时异常的情况。

    2. Spring是如何管理bean的?

      Spring会读取配置文件或者注解定义的bean信息,实例化为一个BeanDefinition,存入到一个List中,然后利用反射实例化对象,存入一个Map<String,Object>中

    3. Bean的生命周期中的扩展点都有哪些?

      • Spring实例化
      • 依赖注入
      • 各种XXAware接口,注入相关实例
      • BeanPostProcessor
      • InitializingBean
      • DisposableBean
    4. Spring常见的注入方式有哪些?

      • 构造函数
      • set方法
      • 注解注入
    5. Spring是如何解决循环依赖的问题的?

      什么是循环依赖?循环依赖分为两种,一种是构造器依赖,这种JVM会报错;另一种是属性依赖。Spring解决的是属性循环依赖。
      Spring是通过递归来实例化bean和它依赖的bean的,直到bean中没有依赖就会返回,然后反递归层层设置上属性。这里有个关键点,如果是循环依赖,就会把没有真实完成的类注入到对应类中。

    6. 哪几种情况会引起事务失效?

      • this引用方法
      • private或者protected

    Springboot

    1. 什么是Springboot?

      Springboot简化了使用Spring的难度,使开发者能快速上手,具体表现在:

      • 节省了繁重的配置
      • 提供了各种starter启动器
    2. Spring Boot 的核心注解是哪个?它主要由哪几个注解组成的?

      SpringbootApplicaition是Springboot的启动类注解,也是核心注解,它是由下面三个注解组成的:

      • ComponentScan:组件扫描,将组件注册到容器中
      • EnableAutoConfiguration:打开自动配置,根据类路径中jar包是否存在来决定是否开启某个默认配置,用exclude可以关闭某个默认配置
      • SpringBootConfiguration: 相当于Configuration,是一个配置类,会被ComponentScan注解扫描到
    3. 什么是 JavaConfig?

      JavaConfig是在 Spring 3.0 开始从一个独立的项目并入到 Spring 中的,通过纯Java方法来配置Springg容器,可以避免使用XML。

    4. Spring Boot 自动配置原理是什么?

      • Springboot启动的时候会通过@EnableAutoConfiguration注解开启自动配置,并找到META-INF/spring.factories中的所有自动配置类,并对其进行加载。
      • 自动配置类都是以AutoConfiguration结尾的,里面包含一个关键性注解EnableConfigurationProperties,这个注解可以把配置了ConfigurationProperties注解的类注入到容器中,这个类可以接收application.properties中配置的配置信息。
    5. YML配置文件的优势在哪里?

      yml格式的配置文件有着可读性更强的树型结构,但是因为yml配置文件需要过多的考虑格式问题,我们项目组还是使用properties配置文件。

    6. Spring Boot 是否可以使用 XML 配置 ?

      可以使用,但是Springboot推荐使用java配置。如果一定要使用的话,可以使用@importResource注解引入一个XML配置

    7. 什么是 Spring Profiles?

      我们的环境分为开发验证和生产环境,没有profile的概念时,需要手动维护多个配置文件,然后进行替换,使用Profile时,可以定义单个配置文件,并定义名称为application-dev.properties,application-verify.properties和application-prod.properties,在总配置文件中指定spring.active.profile为对应的配置文件后缀即可。同样的道理,也可以基于Profile注解实现不同环境注入不同的bean到容器中。

    8. 如何重新加载 Spring Boot 上的更改,而无需重新启动服务器?Spring Boot项目如何热部署?

      使用spring-boot-devtools可以实现热部署,具体操作时,需要在idea中开启自动build项目选项,并在配置文件中增加热部署相关的配置项,比如spring.devtools.restart.enabled设置为true。

    9. Spring Boot 中的 starter 到底是什么 ?

      starter的作用简单来说有下面两个:

      • 方便引入相关依赖
      • 自动配置各模块需要的属性,这里的原理和第四个问题,自动配置原理那个问题是一样的
    10. spring-boot-starter-parent 有什么用 ?

      这个是springboot工程的父级依赖,我觉得最核心的就是定义了各种版本号,这样我们在自己的项目中就不需要写版本号了。其他的功能还有比如默认使用Java8,使用UTF-8编码等等。

    11. 不适用spring-boot-starter-parent,如果创建springboot项目?

      一般来说,这种场景可能出现在该模块已经有了一个父模块存在,无法再引入parent,这种场景下可以在父模块中使用dependacyManagement来实现引入sprintboot

    12. Spring Boot 打成的 jar 和普通的 jar 有什么区别 ?

      boot打成的jar不能作为普通的jar被其他项目引用,因为普通可引用的jar解压后就是包名,而Springboot的jar是放在Boot-INF/classes下的。如果一定要引用,那么也可以在pom文件中增加配置,将springboot打成两个jar,一个可执行,一个可引用。SpringBoot打包可执行可依赖的jar包

    13. 运行 Spring Boot 有哪几种方式?

      • 打jar包用java命令或者打war包用容器执行
      • main方法直接执行
      • maven的插件也可以运行: mvn spring-boot:run
    14. 让你写个starter 你会怎么写?

      • 编写自己的Properties配置类和需要提供的核心服务类XXService
      • 编写AutoConfig类。
        • 通过EnableAutoConfiguration开启自动配置并且注入Properties类的实例;通过@Bean注册核心服务类到容器中,构造函数相关参数就是Properties类中提供的
        • 通过Condition类控制相关的条件
      • 在META-INF/spring.factories文件,里面写入key为一个特别长的配置,value为AutoConfig类的全路径名
        参考资料
  • 相关阅读:
    java 根据对象属性排序
    无法初始化SFTP协议。主机是SFTP服务器吗
    Spring IOC 学习(三)IOC容器的依赖注入
    Spring-IOC学习-02 IOC容器初始化
    nginx简单使用
    Spring-IOC学习-01 IOC重要的几个接口定义
    Spring-IOC学习
    Http Service
    C#从入门到放弃--字符串类型转数字类型
    VS系列--快捷键的使用
  • 原文地址:https://www.cnblogs.com/ging/p/13825819.html
Copyright © 2020-2023  润新知