spring面试题
1、什么是Spring框架?
spring是分层 的全栈轻量级开源框架,以Ioc和Aop为核心思想,提供了展现层Springmvc和业务层事务管理器等众多企业级应用技术,还能整合开源世界众多著名三方框架和类库的java开源框架。2、列举一些重要的Spring模块?
Spring是一个分层非常清晰并且依赖关系、职责定位非常明确的轻量级框架。主要包括这几大模块:数据处理模块,Web模块,Aop/Aspect模块,Core Container模块和Test模块。
3、谈谈自己对Spring Ioc和Aop的理解?
1)、Ioc的理解Ioc叫做控制反转,它是一个技术思想,不是一个技术实现。
描述了java开发领域对象的创建和管理的问题
为什么叫控制反转?控制指的是对象的创建和管理的权利,反转指的是控制的权利交给了外部环境(SpringIoc)。
Ioc和DI描述的是同一个事情(对象实例化和依赖关系维护),只不过角度不一样罢了。
Ioc是站在对象的角度,对象实例化和依赖维护的权利交给了(反转)容器。
DI是站在容器的角度,容器会把对象的依赖的其他对象注入。
2)、Aop的理解
Aop叫做面向切面编程,Aop是OOP的延续。OOP使用封装、继承和多台解决了大多数的代码重复问题。
但有些情况处理不了(顶级父类中的多个方法出现相同的代码,称为横切逻辑相同)。
AOP为独辟蹊径日出抽取机制,在不改变原有逻辑情况下,增强代码逻辑,从根本上上解耦。
为什么叫做面向切面编程?切:指横切逻辑,面:横切逻辑一般影响多个方法,每个方法如同一个点,多个点构成面。
4、Spring Aop和AspectJ Aop有什么区别?
SpringAop是运行时增强,是基于代理实现的,AspectJ是编译时增强,是基于字节码操作。5、Spring中的bean的作用域有哪些?
单例:singleton多例:prototype
reuqest:一次request内有效
Session:同一个session有效
global-session:全局session,spring5中已废弃
6、Spring中的单例bean的线程安全问题了解吗?
单例Bean存在线程安全问题,多线程操作同一个单例的成员变量进行写操作时存在线程安全问题。解决:使用将成员变量定义在ThreadLocal中。
7、Spring框架中用到了哪些设计模式?
工厂模式:通过BeanFactory和ApplicationContext创建bean对象代理模式:SpringAop和事务管理
单例模式:Spring中的Bean默认时单例
模板方法:jdbcTemplate等
观察者模式:Spring事件驱动模型就是观察者模式
适配器模式:多种数据库连接使用适配器,SpringMVC使用处理器适配器去获取对应的具体处理器。
8、@Component和@Bean的区别?
作用对象不同,@Component作用于类,@Bean作用于方法@Component注解通常是通过类路径扫描自动装配到Ioc容器的,@Bean注解是在方法中产生,当我们需要是才实例化。
@Bean注解比@Component注解的自定义性更强,而且很多地方只能通过@Bean注解来注册bean。
9、将一个类声明为Spring Bean的注解有哪些?
@Component,@Service,@Repository,@Controller10、Spring事务管理的方式有几种?
声明式事务和编程式事务11、Spring中的事务隔离级别有哪几种?
ISOLATION_DEFAULT使用数据库默认隔离级别,Mysql默认采用可重复读隔离级别,Oracle默认采用读已提交
ISOLATION_READ_UNCOMMITTED
读未提交,最低隔离界别,可能导致脏读,幻读和不可重复读
ISOLATION_READ_COMMITTED
读已提交,允许读取并发事务已经提交的数据,可避免脏读,幻读和不可重复读可能发生
ISOLATION_REPEATABLE_READ
可重复度,可避免脏读,不可重复读,可能发生幻读
ISOLATION_SERIALIZABLE
串行化,最高隔离界别,但是性能低下。
12、Spring事务中有哪几种事务传播行为?
PROPOGATION_REQUIRED如果当前存在事务,则加入该事务,没有则新建
PROPOGATION_SUPPORTS
如果存在事务,加入,不存在,非事务执行
13、BeanFactory和ApplicationContext有什么区别?
BeanFactory是SpringIoc的顶层接口,只定义了基础规范。ApplicationContext是它的子接口,具备BeanFactory的全部功能。BeanFactory是SpringIoc的基础容器,ApplicationContext是容器的高级接口。拥有更多的功能,比如国际化,资源访问等
14、BeanFactory和FactoryBean的区别?
BeanFactory是容器得顶层接口,定义了一些基础行为,是负责生产和管理bean得一个工厂。FactoryBean是Spring中得一种特殊的工厂Bean,可以生成某一类型的Bean实例,也就是说,可以使用Factory Bean自定义Bean的创建过程。
FactoryBean接口方法
T getObject() 返回FactoryBean创建Bean的实例,如果isSingleton返回true,降入放入单例池。
getObjectType() 返回FactoryBean创建的Bean类型
isSingleton() 返回作用于是否单例
15、描述一下Spring Ioc的初始化过程?
(1)、源码
(2)、初始化过程分析
第一步:调用prepareRefresh()做刷新前的预处理工作第二步:调用obtainFreshBeanFactory()获取BeanFactory,默认实现时DefaultListableBeanFactory,加载BeanDefinition并注册到BeanDefinitionRegist
ry中
第三步:调用prepareBeanFactory()方法对BeanFactory做一些前置准备工作,比如设置context的类加载器等
第四步:调用postProcessorBeanFactory对BeanFactory做一些后置处理
第五步:调用invokeBeanFactoryPostProcessors()使用反射技术调用实现了BeanFactoryPostProcessor接口的Bean
第六步:调用registerBeanPostProcessors()注册BeanPostProcessor(Bean的后置处理器),在Bean的创建前后执行
第七步:调用initMessageSource()初始化MessageSource(做国际化、消息绑定,消息解析等)
第八步:调用initApplicationEventMulticaster()初始化事件派发器
第九步:调用onRefresh()方法,子类将会重写该方法,在刷新时可以自定义逻辑
第十步:调用registerListeners()注册实现了ApplicationListener接口的Bean
第十一步:调用finishBeanFactoryInitialization()实例化所有没有设置懒加载的单例Bean 比如填充属性,调用初始化方法(比如调用afterPropertiesSet(),init-method等),调用BeanPostProcessor对Bean进行后置处理
第十二步:调用finishRefresh()完成对context的刷新
16、谈一谈BeanFactory是如何创建的?
第一步:调用refreshBeanFactory()刷新BeanFactory调用AbstractRefreshableApplicationContext的createBeanFactory()创建 BeanFactory
1、判断是否已经有了BeanFactory,有则销毁Bean并关闭BeanFactory
2、没有则调用createBeanFactory()方法创建BeenFactory并设置一些属性
3、调用loadBeanDefinitions()加载应用中的BeanDefinition,会一直调用到XmlBeanDefinitionReader的doLoadBeanDefinitions()方法
1、调用doLoadDocument()方法读取xml,将xml中的信息保存到Document中
2、调用registerBeanDefinitions()解析document对象,封装BeanDefinition并进行注册
1、一直调用下去会调用到DefaultBeanDefinitionDocumentReader的doRegisterBeanDefinitions()方法
2、继续调用parseBeanDefinitions()方法后再调用parseDefaultElement()方法
3、根据Xml的节点名称为bean后调用processBeanDefinition方法进行解析
1、解析为BeanDefinitionHolder对象,是一个BeanDefinition的包装类型
2、使用BeanDefinitionReaderUtils工具类完成BeanDefinition的注册
第二步:调用getBeanFactorty()获取BeanFactory,实际返回一个DefaultListableBeanFactory
17、谈一谈Spring Bean是如何创建的?
第一步:进入入口方法AbstractApplicationContext的finishBeanFactoryInitialization()方法第二步:调用DefaultListableBeanFactory(也就是上面obtainBeanFactory()的返回值)的preInstantiateSingletons()方法
第三步:不管是工厂Bean还是普通Bean,最终都是通过getBean方法去获取实例
第四步:调用AbstractBeanFactory的doGetBean()去获取Bean实例,调用createBean()方法
第五步:调用AbstractAutowireCapableBeanFactory的createBean方法,接着调用doCreateBean方法
第六步:调用createBeanInstance()创建Bean实例,但是没有进行属性设置(依赖注入)
第七步:调用populateBean()方法进行属性填充,也就是实现依赖注入
第八步:调用initializeBean()进行初始化,并调用BeanPostProcessor后置处理器
18、谈一谈Spring Bean的延迟加载的原理?
普通Bean在容器启动时就初始化了,而被lazy-init=true修饰的Bean则在第一次获取Bean是触发初始化。spring启动时会把所有的bean信息(xml和注解)解析为BeanDefinition存储到HashMap中,然后对每个BeanDefinition进行处理,如果是懒加载则初始化阶段不处理。
在bean初始化过程中的preInstantiateSingltons()方法中,有判断,必须是单例的并且不时懒加载的才会在容器创建时初始化
19、谈一谈Spring Ioc的循环依赖问题?
Spring中的循环依赖有构造器的循环依赖和set注入的循环依赖,构造器循环依赖是没有办法解决的,只能抛出BeanCurrentlyInCreationException异常多实例的Bean的循环依赖也是没有办法解决的,这里说的是单实例set注入的循环依赖解决
Spring的循环依赖的理论依据是基于Java的引用传递,当获取对象的引用时,对象的属性是可以延时设置的,但是构造器必须在获取引用之前。
Spring采用提前暴露一个ObjectFactory对象来完成的,简单说就是ClassA调用setClassB之前就把ClassA实例化的对象通过ObjectFactory提前暴露在Spring容器中了
SpringMVC面试题
1、什么是Springmvc?简单介绍一下你对Springmvc的理解?
Springmvc是一个基于Java的实现了MVC模式的请求驱动类型的轻量级Web框架,通过把Model,VIEW,Controller分离,将web层进行解耦,方便组内开发人员分工合作2、Springmvc的工作流程?
1)、工作流程图
2)、工作流程说明
第一步:用户发送请求到前端控制器DispatchServlet第二步:前端控制器收到请求调用处理器映射器HandlerMapping
第三步:处理器映射器根据URL找到具体的处理器Handler,生成处理器对象以及处理器拦截器链,并返回前端控制器
第四步:前端控制器调用处理器适配器HandlerAdapter去调用处理器Handler
第五步:处理器适配器执行处理器Handler
第六步:处理器Handler执行完成后给处理器适配器返回ModelAndView
第七步:处理器适配器将ModelAndView(包含Model,View)返回给前端控制器
第八步:前端控制器请求视图解析器ViewResolver进行视图解析,根据逻辑视图名解析真正的视图
第九步:视图解析器返回View对象给前端控制器
第十步:前端控制器进行视图渲染,就是将模型数据填充到request域
第十一步:前端控制器向用户响应结果
3、Springmvc的主要组件?
HandlerMapping(处理器映射器)HandlerMapping的作用就是根据URL获取响应的处理器,
标注了@RequestMapping的每个方法都可以看做一个Handler,Handler负责处理具体的请求。
HandlerAdapter(处理器适配器)
HandlerAdapter是一个适配器,SpringMvc中的Handler可以是任意形式,只要能处理请求即可。但是请求交给Servlet时,Servlet方法结构是doService(HttpServletRequest req,HttpServletResponse resp)形式的,要让固定的Servlet方法调用Handler进行处理,便是HandlerAdapter的职责
HandlerExceptionResolver(处理器异常解析器)
用于处理Handler产生的异常
ViewResolver(视图解析器)
用于将String类型的视图名和Local解析为View类型的视图,只有一个resolveViewName()方法
RequestToViewNameTranslator
有些组件没有设置View,也没有设置ViewName,该组件从请求中获取ViewName
LocalResolver
从请求中解析出Local
ThemeResolver
用于解析主题
MultipartResolver
用于上传请求
FlashMapManager
用于重定向时的参数传递
4、Springmvc如何设置重定向和转发?
转发就是在返回之前面加上forward 重定向就是加上redirect:5、如何解决post请求乱码问题,get请求又如何处理?
post请求乱码解决,在web.xml中配置CharacterEncondingFilter, Get请求可以修改Tomcat编码,第二种是对参数进行重新编码。6、Springmvc的异常处理?
@ControllerAdvice+@ExceptionHandler()注解进行处理Springboot面试题
1、什么是Springboot?
SpringBoot用来简化Spring应用开发,约定大于配置,去繁化简的轻量级框架创建独立的Spring应用程序main方法运行
内嵌Tomcat无需部署war包
简化maven配置,很多依赖不需要指定版本号
自动配置Spring添加对应starter即可完成自动化配置
2、Springboot有哪些优缺点?
优点:独立运行,简化配置,自动配置,无代码生成和XML配置(轻量级),应用监控缺点:太容易上手,但是如果不了解核心技术和流程,一旦遇到问题就会棘手
3、Spring,Springmvc,Springboot有什么区别?
Spring最重要的是IOC和AOP技术思想的具体实现,SpringMVC其实可以说就是封装了Servlet,让Web开发更加容易,
Spring和Springmvc均需要大量的配置,SpringBoot自动配置和启动就是解决它们的问题的
4、SpringBoot是如何实现自动配置的?
第一步:首先分析@SpringBootApplication注解,由三个核心注解组成@SpringConfiguration,@EnableAutoConfiguration,@ComponentScan第二步:分析@SpringBootConfiguration发现就是包装了@Configuration注解,声明为一个配置类
第三步:分析@EnableAutoConfiguration注解,它由@AutoConfigurationPackage注解和@Import(AutoConfigurationImportSelector.class)组成
第四步:@AutoConfigurationPackage是使用@Import(AutoConfigurationPackages.Registrar.class)导入了该类实现自动扫描包的功能
第五步:进入Registrar.class可以看到它实现了ImportBeanDefinitionRegistrar接口的registerBeanDefinitions方法,并传入了包名,默认是入口程序的包名称
第六步:回到@EnableAutoConfiguration的第二个注解@Import(AutoConfigurationImportSelector.class),它实现了ImportSelector接口的selectImports方法将符合条件的标注有@Configuration的类注入SpringIoc中
第七步:importSelector会从META-INFO下的spring-autoconfiguration-metadata,properties中加载配置类
第八步:根据元数据信息使用SpringFactoriesLoader在META-INF/spring.factories中加载EnableAutoConfiguration的值,通过反射技术将标注@Configuration的类加载到Ioc容器中,从而实现自动配置
第九步:@ComponentScan可以扫描指定包下面的Bean
5、Springboot打成jar包和普通的jar有什么区别?
Springboot打的jar可以直接执行,普通的jar不能直接执行6、如何使用Springboot实现异常处理?
@ControllerAdvice + @ExceptionHandler7、Springboot中如何解决跨域问题?
重写WebMvcConfigurer的addCorsMappings方法,或者编写CorsFilter8、Springboot的核心配置文件有哪几个?他们的区别是什么?
application(用于自动配置)和bootstrap(SpringCloud Config,需要在bootstrap中添加连接到配置中心的全局配置文件 一些固定的不能被覆盖的属性 一些加密解密场景)9、Springboot有哪几种读取配置的方式?
@Value,@ConfigurationProperties, @PropertySource, @Enviroment10、为什么导入dependency不需要指定版本?
因为spring-boot-start-parent中继承了spring-boot-dependencies,在这里面实现了版本号的统一管理11、为什么要自定义starter?
一些公共的组件可以只要加入jar包,即可使用,方便组件复用12、如何自定义starter?
第一步:引入spring-boot-autoconfiguration依赖第二步:编写自动配置的配置类,并定义需要加载的Bean
第三步:在resources目录下创建META-INF/spring.factories文件
第四步:key为org.springframework.boot.autoconfiguration.EnableAutoConfiguration, value值为自己定义的自动配置类
13、SpringApplication.run是如何做到启动SpringBoot项目的呢?
第一步:进入run方法,发现是new SpringApplication,然后调用run方法,也就是先进行初始化,然后再启动。第二步:主要是调用setInitializers进行初始化,调用setListeners注册监听器,然后给主入口程序类进行赋值
第三步:执行run方法启动容器
第一步:执行getRunListeners()获取并启动监听器
第二步:执行prepareEnviroment()传入监听器,准备环境
第三步:执行createApplicationContext()创建Spring容器
第四步:执行prepareContext()做一些前置处理
第五步:执行refreshContext进行容器的刷新
第六步:执行listeners.started()执行监听器发送容器启动完毕的通知
第七步:调用callReturns()方法返回容器
全部是自己含泪整理,如有错误,请多指教!