城市直播房产教育博客汽车
投稿投诉
汽车报价
买车新车
博客专栏
专题精品
教育留学
高考读书
房产家居
彩票视频
直播黑猫
投资微博
城市上海
政务旅游

Spring高频面试题总结干货(建议收藏)

7月7日 罗刹长投稿
  文章目录:
  Spring的优点Spring用到了哪些设计模式?什么是AOP?AOP有哪些实现方式?JDK动态代理和CGLIB动态代理的区别?SpringAOP相关术语Spring通知有哪些类型?什么是IOC?IOC的优点是什么?什么是依赖注入?IOC容器初始化过程?Bean的生命周期BeanFactory和FactoryBean的区别?Bean注入容器有哪些方式?Bean的作用域Spring自动装配的方式有哪些?Autowired和Resource的区别?Qualifier注解有什么作用Bean和Component有什么区别?Component、Controller、Repositor和Service的区别?Spring事务实现方式有哪些?有哪些事务传播行为?Spring怎么解决循环依赖的问题?Spring启动过程Spring的单例Bean是否有线程安全问题?
  Spring的优点
  轻量,基本版本大约2MB。通过控制反转和依赖注入实现松耦合。支持面向切面的编程,并且把应用业务逻辑和系统服务分开。通过切面和模板减少样板式代码。方便集成各种优秀框架。内部提供了对各种优秀框架的直接支持(如:Hibernate、MyBatis等)。方便程序的测试。Spring支持Junit4,添加注解便可以测试Spring程序。
  Spring用到了哪些设计模式?
  1、简单工厂模式:BeanFactory就是简单工厂模式的体现,根据传入一个唯一标识来获得Bean对象。
  复制代码:OverridepublicObjectgetBean(Stringname)throwsBeansException{assertBeanFactoryActive();returngetBeanFactory()。getBean(name);}
  2、工厂方法模式:FactoryBean就是典型的工厂方法模式。spring在使用getBean()调用获得该bean时,会自动调用该bean的getObject()方法。每个Bean都会对应一个FactoryBean,如SqlSessionFactory对应SqlSessionFactoryBean。
  3、单例模式:一个类仅有一个实例,提供一个访问它的全局访问点。Spring创建Bean实例默认是单例的。
  4、适配器模式:SpringMVC中的适配器HandlerAdatper。由于应用会有多个Controller实现,如果需要直接调用Controller方法,那么需要先判断是由哪一个Controller处理请求,然后调用相应的方法。当增加新的Controller,需要修改原来的逻辑,违反了开闭原则(对修改关闭,对扩展开放)。
  为此,Spring提供了一个适配器接口,每一种Controller对应一种HandlerAdapter实现类,当请求过来,SpringMVC会调用getHandler()获取相应的Controller,然后获取该Controller对应的HandlerAdapter,最后调用HandlerAdapter的handle()方法处理请求,实际上调用的是Controller的handleRequest()。每次添加新的Controller时,只需要增加一个适配器类就可以,无需修改原有的逻辑。
  常用的处理器适配器:SimpleControllerHandlerAdapter,HttpRequestHandlerAdapter,AnnotationMethodHandlerAdapter。
  复制代码:Determinehandlerforthecurrentrequest。mappedHandlergetHandler(processedRequest);HandlerAdapterhagetHandlerAdapter(mappedHandler。getHandler());Actuallyinvokethehandler。mvha。handle(processedRequest,response,mappedHandler。getHandler());publicclassHttpRequestHandlerAdapterimplementsHandlerAdapter{Overridepublicbooleansupports(Objecthandler){handler是被适配的对象,这里使用的是对象的适配器模式return(handlerinstanceofHttpRequestHandler);}OverrideNullablepublicModelAndViewhandle(HttpServletRequestrequest,HttpServletResponseresponse,Objecthandler)throwsException{((HttpRequestHandler)handler)。handleRequest(request,response);}}
  5、代理模式:spring的aop使用了动态代理,有两种方式JdkDynamicAopProxy和Cglib2AopProxy。
  6、观察者模式:spring中observer模式常用的地方是listener的实现,如ApplicationListener。
  7、模板模式:Spring中jdbcTemplate、hibernateTemplate等,就使用到了模板模式。
  什么是AOP?
  面向切面编程,作为面向对象的一种补充,将公共逻辑(事务管理、日志、缓存等)封装成切面,跟业务代码进行分离,可以减少系统的重复代码和降低模块之间的耦合度。切面就是那些与业务无关,但所有业务模块都会调用的公共逻辑。
  AOP有哪些实现方式?
  AOP有两种实现方式:静态代理和动态代理。
  静态代理
  静态代理:代理类在编译阶段生成,在编译阶段将通知织入Java字节码中,也称编译时增强。AspectJ使用的是静态代理。
  缺点:代理对象需要与目标对象实现一样的接口,并且实现接口的方有冗余代码。同时,一旦接口增加方法,目标对象与代理对象都要维护。
  动态代理
  动态代理:代理类在程序运行时创建,AOP框架不会去修改字节码,而是在内存中临时生成一个代理对象,在运行期间对业务方法进行增强,不会生成新类。
  JDK动态代理和CGLIB动态代理的区别?
  SpringAOP中的动态代理主要有两种方式:JDK动态代理和CGLIB动态代理。
  JDK动态代理
  如果目标类实现了接口,SpringAOP会选择使用JDK动态代理目标类。代理类根据目标类实现的接口动态生成,不需要自己编写,生成的动态代理类和目标类都实现相同的接口。JDK动态代理的核心是InvocationHandler接口和Proxy类。
  缺点:目标类必须有实现的接口。如果某个类没有实现接口,那么这个类就不能用JDK动态代理。
  CGLIB动态代理
  通过继承实现。如果目标类没有实现接口,那么SpringAOP会选择使用CGLIB来动态代理目标类。CGLIB(CodeGenerationLibrary)可以在运行时动态生成类的字节码,动态创建目标类的子类对象,在子类对象中增强目标类。
  CGLIB是通过继承的方式做的动态代理,因此如果某个类被标记为final,那么它是无法使用CGLIB做动态代理的。
  优点:目标类不需要实现特定的接口,更加灵活。
  什么时候采用哪种动态代理?
  如果目标对象实现了接口,默认情况下会采用JDK的动态代理实现AOP如果目标对象实现了接口,可以强制使用CGLIB实现AOP如果目标对象没有实现了接口,必须采用CGLIB库
  两者的区别:
  jdk动态代理使用jdk中的类Proxy来创建代理对象,它使用反射技术来实现,不需要导入其他依赖。cglib需要引入相关依赖:asm。jar,它使用字节码增强技术来实现。当目标类实现了接口的时候SpringAop默认使用jdk动态代理方式来增强方法,没有实现接口的时候使用cglib动态代理方式增强方法。
  SpringAOP相关术语
  (1)切面(Aspect):切面是通知和切点的结合。通知和切点共同定义了切面的全部内容。
  (2)连接点(Joinpoint):指方法,在SpringAOP中,一个连接点总是代表一个方法的执行。连接点是在应用执行过程中能够插入切面的一个点。这个点可以是调用方法时、抛出异常时、甚至修改一个字段时。切面代码可以利用这些点插入到应用的正常流程之中,并添加新的行为。
  (3)通知(Advice):在AOP术语中,切面的工作被称为通知。
  (4)切入点(Pointcut):切点的定义会匹配通知所要织入的一个或多个连接点。我们通常使用明确的类和方法名称,或是利用正则表达式定义所匹配的类和方法名称来指定这些切点。
  (5)引入(Introduction):引入允许我们向现有类添加新方法或属性。
  (6)目标对象(TargetObject):被一个或者多个切面(aspect)所通知(advise)的对象。它通常是一个代理对象。
  (7)织入(Weaving):织入是把切面应用到目标对象并创建新的代理对象的过程。在目标对象的生命周期里有以下时间点可以进行织入:
  编译期:切面在目标类编译时被织入。AspectJ的织入编译器是以这种方式织入切面的。类加载期:切面在目标类加载到JVM时被织入。需要特殊的类加载器,它可以在目标类被引入应用之前增强该目标类的字节码。AspectJ5的加载时织入就支持以这种方式织入切面。运行期:切面在应用运行的某个时刻被织入。一般情况下,在织入切面时,AOP容器会为目标对象动态地创建一个代理对象。SpringAOP就是以这种方式织入切面。
  Spring通知有哪些类型?
  在AOP术语中,切面的工作被称为通知。通知实际上是程序运行时要通过SpringAOP框架来触发的代码段。
  Spring切面可以应用5种类型的通知:
  前置通知(Before):在目标方法被调用之前调用通知功能;后置通知(After):在目标方法完成之后调用通知,此时不会关心方法的输出是什么;返回通知(Afterreturning):在目标方法成功执行之后调用通知;异常通知(Afterthrowing):在目标方法抛出异常后调用通知;环绕通知(Around):通知包裹了被通知的方法,在被通知的方法调用之前和调用之后执行自定义的逻辑。
  什么是IOC?
  IOC:控制反转,由Spring容器管理bean的整个生命周期。通过反射实现对其他对象的控制,包括初始化、创建、销毁等,解放手动创建对象的过程,同时降低类之间的耦合度。
  IOC的好处:降低了类之间的耦合,对象创建和初始化交给Spring容器管理,在需要的时候只需向容器进行申请。
  IOC的优点是什么?
  IOC和依赖注入降低了应用的代码量。松耦合。支持加载服务时的饿汉式初始化和懒加载。
  什么是依赖注入?
  在Spring创建对象的过程中,把对象依赖的属性注入到对象中。依赖注入主要有两种方式:构造器注入和属性注入。
  IOC容器初始化过程?
  ioc容器初始化过程:BeanDefinition的资源定位、解析和注册。
  从XML中读取配置文件。将bean标签解析成BeanDefinition,如解析property元素,并注入到BeanDefinition实例中。将BeanDefinition注册到容器BeanDefinitionMap中。BeanFactory根据BeanDefinition的定义信息创建实例化和初始化bean。
  单例bean的初始化以及依赖注入一般都在容器初始化阶段进行,只有懒加载(lazyinit为true)的单例bean是在应用第一次调用getBean()时进行初始化和依赖注入。AbstractApplicationContextInstantiateallremaining(nonlazyinit)singletons。finishBeanFactoryInitialization(beanFactory);
  多例bean在容器启动时不实例化,即使设置lazyinit为false也没用,只有调用了getBean()才进行实例化。
  loadBeanDefinitions采用了模板模式,具体加载BeanDefinition的逻辑由各个子类完成。
  Bean的生命周期
  1。对Bean进行实例化
  2。依赖注入
  3。如果Bean实现了BeanNameAware接口,Spring将调用setBeanName(),设置Bean的id(xml文件中bean标签的id)
  4。如果Bean实现了BeanFactoryAware接口,Spring将调用setBeanFactory()
  5。如果Bean实现了ApplicationContextAware接口,Spring容器将调用setApplicationContext()
  6。如果存在BeanPostProcessor,Spring将调用它们的postProcessBeforeInitialization(预初始化)方法,在Bean初始化前对其进行处理
  7。如果Bean实现了InitializingBean接口,Spring将调用它的afterPropertiesSet方法,然后调用xml定义的initmethod方法,两个方法作用类似,都是在初始化bean的时候执行
  8。如果存在BeanPostProcessor,Spring将调用它们的postProcessAfterInitialization(后初始化)方法,在Bean初始化后对其进行处理
  9。Bean初始化完成,供应用使用,直到应用被销毁
  10。如果Bean实现了DisposableBean接口,Spring将调用它的destory方法,然后调用在xml中定义的destorymethod方法,这两个方法作用类似,都是在Bean实例销毁前执行。publicinterfaceBeanPostProcessor{NullabledefaultObjectpostProcessBeforeInitialization(Objectbean,StringbeanName)throwsBeansException{}NullabledefaultObjectpostProcessAfterInitialization(Objectbean,StringbeanName)throwsBeansException{}}publicinterfaceInitializingBean{voidafterPropertiesSet()throwsE}BeanFactory和FactoryBean的区别?
  BeanFactory:管理Bean的容器,Spring中生成的Bean都是由这个接口的实现来管理的。
  FactoryBean:通常是用来创建比较复杂的bean,一般的bean直接用xml配置即可,但如果一个bean的创建过程中涉及到很多其他的bean和复杂的逻辑,直接用xml配置比较麻烦,这时可以考虑用FactoryBean,可以隐藏实例化复杂Bean的细节。
  当配置文件中bean标签的class属性配置的实现类是FactoryBean时,通过getBean()方法返回的不是FactoryBean本身,而是调用FactoryBeangetObject()方法所返回的对象,相当于FactoryBeangetObject()代理了getBean()方法。如果想得到FactoryBean必须使用beanName的方式获取。
  Mybatis提供了SqlSessionFactoryBean,可以简化SqlSessionFactory的配置:publicclassSqlSessionFactoryBeanimplementsFactoryBeanSqlSessionFactory,InitializingBean,ApplicationListener{OverridepublicvoidafterPropertiesSet()throwsException{notNull(dataSource,PropertydataSourceisrequired);notNull(sqlSessionFactoryBuilder,PropertysqlSessionFactoryBuilderisrequired);state((configurationnullconfigLocationnull)!(configuration!nullconfigLocation!null),PropertyconfigurationandconfigLocationcannotspecifiedwithtogether);this。sqlSessionFactorybuildSqlSessionFactory();}protectedSqlSessionFactorybuildSqlSessionFactory()throwsIOException{复杂逻辑}OverridepublicSqlSessionFactorygetObject()throwsException{if(this。sqlSessionFactorynull){afterPropertiesSet();}returnthis。sqlSessionF}}
  在xml配置SqlSessionFactoryBean:beanidtradeSqlSessionFactoryclassorg。mybatis。spring。SqlSessionFactoryBeanpropertynamedataSourcereftradepropertynamemapperLocationsvalueclasspath:mappertradeMapper。xmlpropertynameconfigLocationvalueclasspath:mybatisconfig。xmlpropertynametypeAliasesPackagevaluecom。bytebeats。mybatis3。domain。tradebean
  Spring将会在应用启动时创建SqlSessionFactory,并使用sqlSessionFactory这个名字存储起来。
  Bean注入容器有哪些方式?
  将普通类交给Spring容器管理,通常有以下方法:
  1、使用Configuration与Bean注解
  2、使用Controller、Service、Repository、Component注解标注该类,然后启用ComponentScan自动扫描
  3、使用Import方法。使用Import注解把bean导入到当前容器中,代码如下:SpringBootApplicationComponentScan把用到的资源导入到当前容器中Import({Dog。class,Cat。class})publicclassApp{publicstaticvoidmain(String〔〕args)throwsException{ConfigurableApplicationContextcontextSpringApplication。run(App。class,args);System。out。println(context。getBean(Dog。class));System。out。println(context。getBean(Cat。class));context。close();}}Bean的作用域
  1、singleton:单例,Spring中的bean默认都是单例的。
  2、prototype:每次请求都会创建一个新的bean实例。
  3、request:每一次HTTP请求都会产生一个新的bean,该bean仅在当前HTTPrequest内有效。
  4、session:每一次HTTP请求都会产生一个新的bean,该bean仅在当前HTTPsession内有效。
  5、globalsession:全局session作用域。
  Spring自动装配的方式有哪些?
  Spring的自动装配有三种模式:byType(根据类型),byName(根据名称)、constructor(根据构造函数)。
  byType
  找到与依赖类型相同的bean注入到另外的bean中,这个过程需要借助setter注入来完成,因此必须存在set方法,否则注入失败。
  当xml文件中存在多个相同类型名称不同的实例Bean时,Spring容器依赖注入仍然会失败,因为存在多种适合的选项,Spring容器无法知道该注入那种,此时我们需要为Spring容器提供帮助,指定注入那个Bean实例。可以通过bean标签的autowirecandidate设置为false来过滤那些不需要注入的实例BeanbeaniduserDaoclasscom。zejian。spring。springIoc。dao。impl。UserDaoImpl!autowirecandidatefalse过滤该类型beaniduserDao2autowirecandidatefalseclasscom。zejian。spring。springIoc。dao。impl。UserDaoImpl!byType根据类型自动装配userDaobeaniduserServiceautowirebyTypeclasscom。zejian。spring。springIoc。service。impl。UserServiceImpl
  byName
  将属性名与bean名称进行匹配,如果找到则注入依赖bean。beaniduserDaoclasscom。zejian。spring。springIoc。dao。impl。UserDaoImplbeaniduserDao2classcom。zejian。spring。springIoc。dao。impl。UserDaoImpl!byName根据名称自动装配,找到UserServiceImpl名为userDao属性并注入beaniduserServiceautowirebyNameclasscom。zejian。spring。springIoc。service。impl。UserServiceImpl
  constructor
  存在单个实例则优先按类型进行参数匹配(无论名称是否匹配),当存在多个类型相同实例时,按名称优先匹配,如果没有找到对应名称,则注入失败,此时可以使用autowirecandidatefalse过滤来解决。!只存在userDao2,userDao3无法成功注入beaniduserDao2classcom。zejian。spring。springIoc。dao。impl。UserDaoImplbeaniduserDao3classcom。zejian。spring。springIoc。dao。impl。UserDaoImplbeaniduserServiceautowireconstructorclasscom。zejian。spring。springIoc。service。impl。UserServiceImpl
  Autowired可以传递了一个requiredfalse的属性,false指明当userDao实例存在就注入不存就忽略,如果为true,就必须注入,若userDao实例不存在,就抛出异常。由于默认情况下Autowired是按类型匹配的(byType),如果需要按名称(byName)匹配的话,可以使用Qualifier注解与Autowired结合。publicclassUserServiceImplimplementsUserService{标注成员变量AutowiredQualifier(userDao1)privateUserDaouserD}
  byName模式xml配置:!根据Qualifier(userDao1)自动识别beaniduserDao1classcom。zejian。spring。springIoc。dao。impl。UserDaoImplbeaniduserDao2classcom。zejian。spring。springIoc。dao。impl。UserDaoImplbeaniduserServiceclasscom。zejian。spring。springIoc。service。impl。UserServiceImpl
  Resource,默认按byName模式自动注入。Resource有两个中重要的属性:name和type。Spring容器对于Resource注解的name属性解析为bean的名字,type属性则解析为bean的类型。因此使用name属性,则按byName模式的自动注入策略,如果使用type属性则按byType模式自动注入策略。倘若既不指定name也不指定type属性,Spring容器将通过反射技术默认按byName模式注入。Resource(nameuserDao)privateUserDaouserD用于成员变量也可以用于set方法标注Resource(nameuserDao)publicvoidsetUserDao(UserDaouserDao){this。userDaouserD}
  上述两种自动装配的依赖注入并不适合简单值类型,如int、boolean、long、String以及Enum等,对于这些类型,Spring容器也提供了Value注入的方式。Value接收一个String的值,该值指定了将要被注入到内置的java类型属性值,Spring容器会做好类型转换。一般情况下Value会与properties文件结合使用。
  jdbc。properties文件如下:jdbc。drivercom。mysql。jdbc。Driverjdbc。urljdbc:mysql:127。0。0。1:3306test?characterEncodingUTF8allowMultiQueriestruejdbc。usernamerootjdbc。passwordroot
  利用注解Value获取jdbc。url和jdbc。username的值,实现如下:publicclassUserServiceImplimplementsUserService{占位符方式Value({jdbc。url})privateSSpEL表达方式,其中代表xml配置文件中的id值configPropertiesValue({configProperties〔jdbc。username〕})privateStringuserN}
  xml配置文件:!基于占位符方式配置单个properties!context:propertyplaceholderlocationconfjdbc。properties!基于占位符方式配置多个propertiesbeanidpropertyConfigurerclassorg。springframework。beans。factory。config。PreferencesPlaceholderConfigurerpropertynamelocationvalueconfjdbc。propertiesbean!基于SpEL表达式配置多个propertiesid值为configProperties提供java代码中使用beanidconfigPropertiesclassorg。springframework。beans。factory。config。PropertiesFactoryBeanpropertynamelocationslistvalueclasspath:confjdbc。propertiesvaluelistpropertybeanAutowired和Resource的区别?
  Autowired注解是按照类型(byType)装配依赖对象的,但是存在多个类型致的bean,法通过byType注时,就会再使byName来注,如果还是法判断注哪个bean则会UnsatisfiedDependencyException。
  Resource会先按照byName来装配,如果找不到bean,会动byType再找次。
  Qualifier注解有什么作用
  当需要创建多个相同类型的bean并希望仅使用属性装配其中一个bean时,可以使用Qualifier注解和Autowired通过指定应该装配哪个bean来消除歧义。
  Bean和Component有什么区别?
  都是使用注解定义Bean。Bean是使用Java代码装配Bean,Component是自动装配Bean。
  Component注解用在类上,表明一个类会作为组件类,并告知Spring要为这个类创建bean,每个类对应一个Bean。
  Bean注解用在方法上,表示这个方返回一个Bean。Bean需要在配置类中使用,即类上需要加上Configuration注解。ComponentpublicclassStudent{privateSpublicStringgetName(){}}ConfigurationpublicclassWebSocketConfig{BeanpublicStudentstudent(){returnnewStudent();}}
  Bean注解更加灵活。当需要将第三方类装配到Spring容器中,因为没办法源代码上添加Component注解,只能使用Bean注解的方式,当然也可以使用xml的方式。
  Component、Controller、Repositor和Service的区别?
  Component:最普通的组件,可以被注入到spring容器进行管理。
  Controller:将类标记为SpringWebMVC控制器。
  Service:将类标记为业务层组件。
  Repository:将类标记为数据访问组件,即DAO组件。
  Spring事务实现方式有哪些?
  事务就是一系列的操作原子执行。Spring事务机制主要包括声明式事务和编程式事务。
  编程式事务:通过编程的方式管理事务,这种方式带来了很大的灵活性,但很难维护。声明式事务:将事务管理代码从业务方法中分离出来,通过aop进行封装。Spring声明式事务使得我们无需要去处理获得连接、关闭连接、事务提交和回滚等这些操作。使用Transactional注解开启声明式事务。
  Transactional相关属性如下:
  有哪些事务传播行为?
  在TransactionDefinition接口中定义了七个事务传播行为:
  PROPAGATIONREQUIRED如果存在一个事务,则支持当前事务。如果没有事务则开启一个新的事务。如果嵌套调用的两个方法都加了事务注解,并且运行在相同线程中,则这两个方法使用相同的事务中。如果运行在不同线程中,则会开启新的事务。PROPAGATIONSUPPORTS如果存在一个事务,支持当前事务。如果没有事务,则非事务的执行。PROPAGATIONMANDATORY如果已经存在一个事务,支持当前事务。如果不存在事务,则抛出异常IllegalTransactionStateException。PROPAGATIONREQUIRESNEW总是开启一个新的事务。需要使用JtaTransactionManager作为事务管理器。PROPAGATIONNOTSUPPORTED总是非事务地执行,并挂起任何存在的事务。需要使用JtaTransactionManager作为事务管理器。PROPAGATIONNEVER总是非事务地执行,如果存在一个活动事务,则抛出异常。PROPAGATIONNESTED如果一个活动的事务存在,则运行在一个嵌套的事务中。如果没有活动事务,则按PROPAGATIONREQUIRED属性执行。
  PROPAGATIONNESTED与PROPAGATIONREQUIRESNEW的区别:
  使用PROPAGATIONREQUIRESNEW时,内层事务与外层事务是两个独立的事务。一旦内层事务进行了提交后,外层事务不能对其进行回滚。两个事务互不影响。
  使用PROPAGATIONNESTED时,外层事务的回滚可以引起内层事务的回滚。而内层事务的异常并不会导致外层事务的回滚,它是一个真正的嵌套事务。
  Spring怎么解决循环依赖的问题?
  构造器注入的循环依赖:Spring处理不了,直接抛出BeanCurrentlylnCreationException异常。
  单例模式下属性注入的循环依赖:通过三级缓存处理循环依赖。
  非单例循环依赖:无法处理。
  下面分析单例模式下属性注入的循环依赖是怎么处理的:
  首先,Spring单例对象的初始化大略分为三步:
  createBeanInstance:实例化bean,使用构造方法创建对象,为对象分配内存。populateBean:进行依赖注入。initializeBean:初始化bean。
  Spring为了解决单例的循环依赖问题,使用了三级缓存:
  singletonObjects:完成了初始化的单例对象map,beannamebeaninstance
  earlySingletonObjects:完成实例化未初始化的单例对象map,beannamebeaninstance
  singletonFactories:单例对象工厂map,beannameObjectFactory,单例对象实例化完成之后会加入singletonFactories。
  在调用createBeanInstance进行实例化之后,会调用addSingletonFactory,将单例对象放到singletonFactories中。protectedvoidaddSingletonFactory(StringbeanName,ObjectF?singletonFactory){Assert。notNull(singletonFactory,Singletonfactorymustnotbenull);synchronized(this。singletonObjects){if(!this。singletonObjects。containsKey(beanName)){this。singletonFactories。put(beanName,singletonFactory);this。earlySingletonObjects。remove(beanName);this。registeredSingletons。add(beanName);}}}
  假如A依赖了B的实例对象,同时B也依赖A的实例对象。
  A首先完成了实例化,并且将自己添加到singletonFactories中接着进行依赖注入,发现自己依赖对象B,此时就尝试去get(B)发现B还没有被实例化,对B进行实例化然后B在初始化的时候发现自己依赖了对象A,于是尝试get(A),尝试一级缓存singletonObjects和二级缓存earlySingletonObjects没找到,尝试三级缓存singletonFactories,由于A初始化时将自己添加到了singletonFactories,所以B可以拿到A对象,然后将A从三级缓存中移到二级缓存中B拿到A对象后顺利完成了初始化,然后将自己放入到一级缓存singletonObjects中此时返回A中,A此时能拿到B的对象顺利完成自己的初始化
  由此看出,属性注入的循环依赖主要是通过将实例化完成的bean添加到singletonFactories来实现的。而使用构造器依赖注入的bean在实例化的时候会进行依赖注入,不会被添加到singletonFactories中。比如A和B都是通过构造器依赖注入,A在调用构造器进行实例化的时候,发现自己依赖B,B没有被实例化,就会对B进行实例化,此时A未实例化完成,不会被添加到singtonFactories。而B依赖于A,B会去三级缓存寻找A对象,发现不存在,于是又会实例化A,A实例化了两次,从而导致抛异常。
  总结:1、利用缓存识别已经遍历过的节点;2、利用Java引用,先提前设置对象地址,后完善对象。
  Spring启动过程
  读取web。xml文件。创建ServletContext,为ioc容器提供宿主环境。触发容器初始化事件,调用contextLoaderListener。contextInitialized()方法,在这个方初始化一个应用上下文WebApplicationContext,即Spring的ioc容器。ioc容器初始化完成之后,会被存储到ServletContext中。初始化web。xml中配置的Servlet。如DispatcherServlet,用于匹配、处理每个servlet请求。
  Spring的单例Bean是否有线程安全问题?
  当多个用户同时请求一个服务时,容器会给每一个请求分配一个线程,这时多个线程会并发执行该请求对应的业务逻辑,如果业务逻辑有对单例状态的修改(体现为此单例的成员属性),则必须考虑线程安全问题。
  若每个线程中对全局变量、静态变量只有读操作,而无写操作,那么不会有线程安全问题;若有多个线程同时执行写操作,一般都需要考虑线程同步,否则就可能影响线程安全。
  无状态bean和有状态bean
  有实例变量的bean,可以保存数据,是非线程安全的。没有实例变量的对象。不能保存数据,是线程安全的。
  在Spring中无状态的Bean适合用单例模式,这样可以共享实例提高性能。有状态的Bean在多线程环境下不安全,一般用Prototype模式或者使用ThreadLocal解决线程安全问题。
  如果本文对你有帮助,别忘记给我个3连,点赞,转发,评论,
  ,咱们下期见!答案获取方式:已赞已评已关
投诉 评论 转载

男人最想背着女人做的事存下私房钱、抽烟有多少事情是男人想背着女人做的,我们一起来看看吧。对于女人的种种假想行为,男人实在有苦难言。实际上他们最想背着女人做什么呢?笔者综合归纳起来,其实不外乎以……曝8K视频拍摄三星Note21Ultra,Note20即将跌曝光推后发布但一定不会缺席三星Note21Ultra新机上下边框是直角设计的,曝光推后发布但一定不会缺席三星Note21Ultra新机曲面弧度比较大,曝光推后发布但一定不会缺席……Spring高频面试题总结干货(建议收藏)文章目录:Spring的优点Spring用到了哪些设计模式?什么是AOP?AOP有哪些实现方式?JDK动态代理和CGLIB动态代理的区别?SpringAOP相关术语Spr……政府资产减值调拨及处置核算问题探析任伟林海云【摘要】政府资产的确认、计量和报告是政府资产核算的重点和难点,如何规范完善政府资产的减值、无偿调拨和处置的核算,对落实政府会计制度具有重要意义。文章选取了实际工……新班主任个人工作总结经过两个月紧张的工作,我深深地体会到作为一名新班主任的责任重大,虽然已将基础工作基本理顺,但仍然在处理突发事件上存在很多欠缺,现将近两个月的工作情况及个人心得进行汇报:一……水浒传第五十四回读后感分享【水浒传每回读后感第五十四回】入云龙斗法破高廉黑旋风探穴救柴进罗真人知道,高廉与公孙胜的法术手段在伯仲之间,为此,罗直人专门教了公孙胜一套五雷天罡正法,这样,公孙胜……年度班主任述职报告多篇多篇【编辑】年度班主任述职报告多篇【多篇】为好范文网的会员kama投稿推荐,但愿对你的学习工作带来帮助。年度班主任述职报告有哪些?本年度我努力探索,大胆尝试,逐渐摸索出了一套……凤琴啊泣别母亲时代的一粒尘埃,落在每个人的身上都是一座山!妈妈走了,在东北故乡的养老院里走了,我却被封控在上海家中动弹不得,打电话到东北的防疫电话永远是忙音。托了那里的亲戚,告诉我那里……迫切需要造句用迫切需要造句大全91由于卧铺客车上乘客承受卧姿振动,与立姿、坐姿人体振动有较大差别,迫切需要客观准确的舒适性评价方法。92随着家电产品的增加,废旧家电产品的回收成为一个迫切需要解决的问题……专家教你攻破面试关面试,是所有人都要经历的求职关。要跨越这道关卡,个中精髓可不能不掌握。近日,液压英才网在线访谈室特邀英才职业顾问联盟专家、赫敦职业顾问公司资深顾问师徐国芳女士面授机宜企业……过高造句用过高造句大全121,造成工植被土壤干化的直接原因是植被类型选择不当,群落密度过大和群落生产力过高。122,我们总是对短期收益期望过高,却对长期收益期望过低。李笑来123,女保镖……冰之恋唇膏使用注意事项时至秋冬,天气越来越冷,空气越来越干燥,很多人的唇部开始干裂起皮,需要润唇膏护唇。虽然很小的一件事,但是依旧有人不能作对。那么,冰之恋唇膏使用注意事项呢?下面就让本站小编来介绍……
以色列朝圣之旅地球岁月的一道皱痕拉蒙大峡谷小学五年级作文有关母爱迎风造句用迎风造句大全恐龙灭绝之谜并非由于奇科苏卢布陨石撞击古代惩罚道具三角木马女性被如此折磨蹭世界杯热度?卡塔尔小王子首次回应身世问题,网友挺抓马的关于校园防火安全的手抄报大全可用性资讯建筑真正战略领导者的个习惯平台造句用平台造句大全人生有六气,请逼自己养成!我和我的祖国想要达到财富自由的阶段,必须要有投资理财的能力吗?腾讯TIM可支持微信登录吗微信登录TIM收得到微信消息吗轨道车辆的营销战略粉丝带录取通知书见郑恺!场面太暖,这才是正确的追星方式外婆皮皮虾剥皮小窍门皮皮虾剥皮方法跟吴京学做成功六年级作文创编故事醉驾超过200最多罚款多少钱?中年时段的情感危机要母乳喂养不能吃什么恐高症产生的原因是什么

友情链接:中准网聚热点快百科快传网快生活快软网快好知文好找