SpringFramework现在几乎已成为JavaWeb开发的标配框架。那么,作为Java程序员,你对Spring的主要技术点又掌握了多少呢?不妨用本文的问题来检测一下。
本文内容主要翻译自Top50SpringInterviewQuestionsYouMustPrepareIn2018
Spring核心容器–该层基本上是SpringFramework的核心。它包含以下模块:
SpringCore SpringBean SpEL(SpringExpressionLanguage) SpringContext数据访问/集成–该层提供与数据库交互的支持。它包含以下模块:
JDBC(JavaDataBaseConnectivity) ORM(ObjectRelationalMapping) OXM(ObjectXMLMappers) JMS(JavaMessagingService) TransactionWeb–该层提供了创建Web应用程序的支持。它包含以下模块:
Web Web–Servlet Web–Socket Web–PortletAOP–该层支持面向切面编程
Instrumentation–该层为类检测和类加载器实现提供支持。
Test–该层为使用JUnit和TestNG进行测试提供支持。
几个杂项模块:
Messaging–该模块为STOMP提供支持。它还支持注解编程模型,该模型用于从WebSocket客户端路由和处理STOMP消息。 Aspects–该模块为与AspectJ的集成提供支持。Spring配置文件是XML文件。该文件主要包含类信息。它描述了这些类是如何配置以及相互引入的。但是,XML配置文件冗长且更加干净。如果没有正确规划和编写,那么在大项目中管理变得非常困难。
Spring应用一般有以下组件:
接口-定义功能。 Bean类-它包含属性,setter和getter方法,函数等。 Spring面向切面编程(AOP)-提供面向切面编程的功能。 Bean配置文件-包含类的信息以及如何配置它们。 用户程序-它使用接口。使用Spring有以下方式:
作为一个成熟的SpringWeb应用程序。 作为第三方Web框架,使用SpringFrameworks中间层。 用于远程使用。 作为企业级JavaBean,它可以包装现有的POJO(PlainOldJavaObjects)。Spring框架的核心是Spring容器。容器创建对象,将它们装配在一起,配置它们并管理它们的完整生命周期。Spring容器使用依赖注入来管理组成应用程序的组件。容器通过读取提供的配置元数据来接收对象进行实例化,配置和组装的指令。该元数据可以通过XML,Java注解或Java代码提供。
在依赖注入中,您不必创建对象,但必须描述如何创建它们。您不是直接在代码中将组件和服务连接在一起,而是描述配置文件中哪些组件需要哪些服务。由IoC容器将它们装配在一起。
通常,依赖注入可以通过三种方式完成,即:
构造函数注入 setter注入 接口注入在SpringFramework中,仅使用构造函数和setter注入。
IoC的一些好处是:
它将最小化应用程序中的代码量。 它将使您的应用程序易于测试,因为它不需要单元测试用例中的任何单例或JNDI查找机制。 它以最小的影响和最少的侵入机制促进松耦合。 它支持即时的实例化和延迟加载服务。Spring中的IoC的实现原理就是工厂模式加反射机制。
示例:
interfaceFruit{ publicabstractvoideat(); } classAppleimplementsFruit{ publicvoideat(){ System.out.println("Apple"); } } classOrangeimplementsFruit{ publicvoideat(){ System.out.println("Orange"); } } classFactory{ publicstaticFruitgetInstance(StringClassName){ Fruitf=null; try{ f=(Fruit)Class.forName(ClassName).newInstance(); }catch(Exceptione){ e.printStackTrace(); } returnf; } } classClient{ publicstaticvoidmain(String[]a){ Fruitf=Factory.getInstance("io.github.dunwu.spring.Apple"); if(f!=null){ f.eat(); } } }bean所需的依赖项和服务在XML格式的配置文件中指定。这些配置文件通常包含许多bean定义和特定于应用程序的配置选项。它们通常以bean标签开头。例如:
<beanid="studentbean"class="org.edureka.firstSpring.StudentBean"> <propertyname="name"value="Edureka"></property> </bean> 基于注解配置您可以通过在相关的类,方法或字段声明上使用注解,将bean配置为组件类本身,而不是使用XML来描述bean装配。默认情况下,Spring容器中未打开注解装配。因此,您需要在使用它之前在Spring配置文件中启用它。例如:
<beans> <context:annotation-config/> <!--beandefinitionsgohere--> </beans> 基于JavaAPI配置Spring的Java配置是通过使用@Bean和@Configuration来实现。
@Bean注解扮演与<bean/>元素相同的角色。 @Configuration类允许通过简单地调用同一个类中的其他@Bean方法来定义bean间依赖关系。例如:
@Configuration publicclassStudentConfig{ @Bean publicStudentBeanmyStudent(){ returnnewStudentBean(); } }Springbean支持5种scope:
Singleton-每个SpringIoC容器仅有一个单实例。 Prototype-每次请求都会产生一个新的实例。 Request-每一次HTTP请求都会产生一个新的实例,并且该bean仅在当前HTTP请求内有效。 Session-每一次HTTP请求都会产生一个新的bean,同时该bean仅在当前HTTPsession内有效。 Global-session-类似于标准的HTTPSession作用域,不过它仅仅在基于portlet的web应用中才有意义。Portlet规范定义了全局Session的概念,它被所有构成某个portletweb应用的各种不同的portlet所共享。在globalsession作用域中定义的bean被限定于全局portletSession的生命周期范围内。如果你在web中使用globalsession作用域来标识bean,那么web会自动当成session类型来使用。仅当用户使用支持Web的ApplicationContext时,最后三个才可用。
springbean容器的生命周期流程如下:
Spring容器根据配置中的bean定义中实例化bean。 Spring使用依赖注入填充所有属性,如bean中所定义的配置。 如果bean实现BeanNameAware接口,则工厂通过传递bean的ID来调用setBeanName()。 如果bean实现BeanFactoryAware接口,工厂通过传递自身的实例来调用setBeanFactory()。 如果存在与bean关联的任何BeanPostProcessors,则调用preProcessBeforeInitialization()方法。 如果为bean指定了init方法(<bean>的init-method属性),那么将调用它。 最后,如果存在与bean关联的任何BeanPostProcessors,则将调用postProcessAfterInitialization()方法。 如果bean实现DisposableBean接口,当spring容器关闭时,会调用destory()。 如果为bean指定了destroy方法(<bean>的destroy-method属性),那么将调用它。只有将bean用作另一个bean的属性时,才能将bean声明为内部bean。为了定义bean,Spring的基于XML的配置元数据在<property>或<constructor-arg>中提供了<bean>元素的使用。内部bean总是匿名的,它们总是作为原型。
例如,假设我们有一个Student类,其中引用了Person类。这里我们将只创建一个Person类实例并在Student中使用它。
Student.java
publicclassStudent{ privatePersonperson; //SettersandGetters } publicclassPerson{ privateStringname; privateStringaddress; //SettersandGetters }bean.xml
<beanid=“StudentBean"class="com.edureka.Student"> <propertyname="person"> <!--Thisisinnerbean--> <beanclass="com.edureka.Person"> <propertyname="name"value=“Scott"></property> <propertyname="address"value=“Bangalore"></property> </bean> </property> </bean>当bean在Spring容器中组合在一起时,它被称为装配或bean装配。Spring容器需要知道需要什么bean以及容器应该如何使用依赖注入来将bean绑定在一起,同时装配bean。
Spring容器能够自动装配bean。也就是说,可以通过检查BeanFactory的内容让Spring自动解析bean的协作者。
自动装配的不同模式:
no-这是默认设置,表示没有自动装配。应使用显式bean引用进行装配。 byName-它根据bean的名称注入对象依赖项。它匹配并装配其属性与XML文件中由相同名称定义的bean。 byType-它根据类型注入对象依赖项。如果属性的类型与XML文件中的一个bean名称匹配,则匹配并装配属性。 构造函数-它通过调用类的构造函数来注入依赖项。它有大量的参数。 autodetect-首先容器尝试通过构造函数使用autowire装配,如果不能,则尝试通过byType自动装配。不使用XML来描述bean装配,开发人员通过在相关的类,方法或字段声明上使用注解将配置移动到组件类本身。它可以作为XML设置的替代方案。例如:
Spring的Java配置是通过使用@Bean和@Configuration来实现。
@Bean注解扮演与元素相同的角色。 @Configuration类允许通过简单地调用同一个类中的其他@Bean方法来定义bean间依赖关系。例如:
@Configuration publicclassStudentConfig{ @Bean publicStudentBeanmyStudent(){ returnnewStudentBean(); } }默认情况下,Spring容器中未打开注解装配。因此,要使用基于注解装配,我们必须通过配置<context:annotation-config/>元素在Spring配置文件中启用它。
@Required应用于bean属性setter方法。此注解仅指示必须在配置时使用bean定义中的显式属性值或使用自动装配填充受影响的bean属性。如果尚未填充受影响的bean属性,则容器将抛出BeanInitializationException。
示例:
publicclassEmployee{ privateStringname; @Required publicvoidsetName(Stringname){ this.name=name; } publicstringgetName(){ returnname; } }@Autowired可以更准确地控制应该在何处以及如何进行自动装配。此注解用于在setter方法,构造函数,具有任意名称或多个参数的属性或方法上自动装配bean。默认情况下,它是类型驱动的注入。
publicclassEmployee{ privateStringname; @Autowired publicvoidsetName(Stringname){ this.name=name; } publicstringgetName(){ returnname; } }当您创建多个相同类型的bean并希望仅使用属性装配其中一个bean时,您可以使用@Qualifier注解和@Autowired通过指定应该装配哪个确切的bean来消除歧义。
例如,这里我们分别有两个类,Employee和EmpAccount。在EmpAccount中,使用@Qualifier指定了必须装配id为emp1的bean。
Employee.java
publicclassEmployee{ privateStringname; @Autowired publicvoidsetName(Stringname){ this.name=name; } publicstringgetName(){ returnname; } }EmpAccount.java
publicclassEmpAccount{ privateEmployeeemp; @Autowired @Qualifier(emp1) publicvoidshowName(){ System.out.println(“Employeename:”+emp.getName); } }@RequestMapping注解用于将特定HTTP请求方法映射到将处理相应请求的控制器中的特定类/方法。此注释可应用于两个级别:
类级别:映射请求的URL 方法级别:映射URL以及HTTP请求方法SpringDAO使得JDBC,Hibernate或JDO这样的数据访问技术更容易以一种统一的方式工作。这使得用户容易在持久性技术之间切换。它还允许您在编写代码时,无需考虑捕获每种技术不同的异常。
我们可以通过两种方式使用Spring访问Hibernate:
使用Hibernate模板和回调进行控制反转 扩展HibernateDAOSupport并应用AOP拦截器节点Spring支持两种类型的事务管理:
程序化事务管理:在此过程中,在编程的帮助下管理事务。它为您提供极大的灵活性,但维护起来非常困难。 声明式事务管理:在此,事务管理与业务代码分离。仅使用注解或基于XML的配置来管理事务。AOP(Aspect-OrientedProgramming),即面向切面编程
,它与OOP(Object-OrientedProgramming,面向对象编程)相辅相成,提供了与OOP不同的抽象软件结构的视角.
在OOP中,我们以类(class)作为我们的基本单元,而AOP中的基本单元是Aspect(切面)aspect由pointcount和advice组成,它既包含了横切逻辑的定义,也包括了连接点的定义.SpringAOP就是负责实施切面的框架,它将切面所定义的横切逻辑编织到切面所指定的连接点中.
AOP的工作重心在于如何将增强编织目标对象的连接点上,这里包含两个工作: 如何通过pointcut和advice定位到特定的joinpoint上 如何在advice中编写切面代码.可以简单地认为,使用@Aspect注解的类就是切面.
程序运行中的一些时间点,例如一个方法的执行,或者是一个异常的处理.
在SpringAOP中,joinpoint总是方法的执行点。
特定JoinPoint处的Aspect所采取的动作称为Advice。SpringAOP使用一个Advice作为拦截器,在JoinPoint“周围”维护一系列的拦截器。
concern是我们想要在应用程序的特定模块中定义的行为。它可以定义为我们想要实现的功能。
cross-cuttingconcern是一个适用于整个应用的行为,这会影响整个应用程序。例如,日志记录,安全性和数据传输是应用程序几乎每个模块都需要关注的问题,因此它们是跨领域的问题。
实现AOP的技术,主要分为两大类:
静态代理-指使用AOP框架提供的命令进行编译,从而在编译阶段就可生成AOP代理类,因此也称为编译时增强;
编译时编织(特殊编译器实现) 类加载时编织(特殊的类加载器实现)。动态代理-在运行时在内存中“临时”生成AOP动态代理类,因此也被称为运行时增强。
JDK动态代理 CGLIBSpringAOP基于动态代理方式实现;AspectJ基于静态代理方式实现。
SpringAOP仅支持方法级别的PointCut;提供了完全的AOP支持,它还支持属性级别的PointCut。将Advice应用于目标对象后创建的对象称为代理。在客户端对象的情况下,目标对象和代理对象是相同的。
Advice+TargetObject=Proxy为了创建一个advice对象而链接一个aspect和其它应用类型或对象,称为编织(Weaving)。在SpringAOP中,编织在运行时执行。请参考下图:
SpringWebMVC框架提供模型-视图-控制器架构和随时可用的组件,用于开发灵活且松散耦合的Web应用程序。MVC模式有助于分离应用程序的不同方面,如输入逻辑,业务逻辑和UI逻辑,同时在所有这些元素之间提供松散耦合。
DispatcherServlet的工作流程可以用一幅图来说明:
向服务器发送HTTP请求,请求被前端控制器DispatcherServlet捕获。
DispatcherServlet根据-servlet.xml中的配置对请求的URL进行解析,得到请求资源标识符(URI)。然后根据该URI,调用HandlerMapping获得该Handler配置的所有相关的对象(包括Handler对象以及Handler对象对应的拦截器),最后以HandlerExecutionChain对象的形式返回。
DispatcherServlet根据获得的Handler,选择一个合适的HandlerAdapter。(附注:如果成功获得HandlerAdapter后,此时将开始执行拦截器的preHandler(...)方法)。
提取Request中的模型数据,填充Handler入参,开始执行Handler(Controller)。在填充Handler的入参过程中,根据你的配置,Spring将帮你做一些额外的工作:
HttpMessageConveter:将请求消息(如Json、xml等数据)转换成一个对象,将对象转换为指定的响应信息。 数据转换:对请求消息进行数据转换。如String转换成Integer、Double等。 数据根式化:对请求消息进行数据格式化。如将字符串转换成格式化数字或格式化日期等。 数据验证:验证数据的有效性(长度、格式等),验证结果存储到BindingResult或Error中。Handler(Controller)执行完成后,向DispatcherServlet返回一个ModelAndView对象;
根据返回的ModelAndView,选择一个适合的ViewResolver(必须是已经注册到Spring容器中的ViewResolver)返回给DispatcherServlet。
ViewResolver结合Model和View,来渲染视图。
视图负责将渲染结果返回给客户端。
WebApplicationContext是ApplicationContext的扩展。它具有Web应用程序所需的一些额外功能。它与普通的ApplicationContext在解析主题和决定与哪个servlet关联的能力方面有所不同。
(完)
?想学习更多Spring内容可以访问我的Spring教程:spring-notes
本文内容总结:Spring面试问题TOP50,1.一般问题,1.1.不同版本的SpringFramework有哪些主要功能?,1.2.什么是SpringFramework?,1.3.列举SpringFramework的优点。,1.4.SpringFramework有哪些不同的功能?,1.5.SpringFramework中有多少个模块,它们分别是什么?,1.6.什么是Spring配置文件?,1.7.Spring应用程序有哪些不同组件?,1.8.使用Spring有哪些方式?,2.依赖注入(Ioc),2.1.什么是SpringIOC容器?,2.2.什么是依赖注入?,2.3.可以通过多少种方式完成依赖注入?,2.4.区分构造函数注入和setter注入。,2.5.spring中有多少种IOC容器?,2.6.区分BeanFactory和ApplicationContext。,2.7.列举IoC的一些好处。,2.8.SpringIoC的实现机制。,3.Beans,3.1.什么是springbean?,3.2.spring提供了哪些配置方式?,3.3.spring支持集中beanscope?,3.4.springbean容器的生命周期是什么样的?,3.5.什么是spring的内部bean?,3.6.什么是spring装配,3.7.自动装配有哪些方式?,3.8.自动装配有什么局限?,4.注解,4.1.什么是基于注解的容器配置,4.2.如何在spring中启动注解装配?,4.3.@Component,@Controller,@Repository,@Service有何区别?,4.4.@Required注解有什么用?,4.5.@Autowired注解有什么用?,4.6.@Qualifier注解有什么用?,4.7.@RequestMapping注解有什么用?,5.数据访问,5.1.springDAO有什么用?,5.2.列举SpringDAO抛出的异常。,5.3.springJDBCAPI中存在哪些类?,5.4.使用Spring访问Hibernate的方法有哪些?,5.5.列举spring支持的事务管理类型,5.6.spring支持哪些ORM框架,6.AOP,6.1.什么是AOP?,6.2.什么是Aspect?,6.3.什么是切点(JoinPoint),6.4.什么是通知(Advice)?,6.5.有哪些类型的通知(Advice)?,6.6.指出在springaop中concern和cross-cuttingconcern的不同之处。,6.7.AOP有哪些实现方式?,6.8.SpringAOPandAspectJAOP有什么区别?,6.9.如何理解Spring中的代理?,6.10.什么是编织(Weaving)?,7.MVC,7.1.SpringMVC框架有什么用?,7.2.描述一下DispatcherServlet的工作流程,7.3.介绍一下WebApplicationContext,8.资料,
原文链接:https://www.cnblogs.com/jingmoxukong/p/9408037.html