Spring

作者:加菲猫 2018-06-19 210 0

Spring 框架的优点

轻量,大约 2 M。

控制反转:松耦合,对象给出它们的依赖,而不是创建或查找依赖的对象。

容器:包含并管理应用中的对象的声明周期和配置。

面向切面编程(AOP):把应用业务逻辑和系统服务分开。

MVC框架:是 Web 框架的一个很好的替代品。

事务管理:Spring 提供一个持续的事务管理接口,可以扩展到上至本地事务下至全局事务(JTA)。

异常处理:Spring 提供方便的 API 把具体技术相关的异常(比如由 JDBC,Hibernate or JDO 抛出的)转化为一致的 unchecked 异常。

SpringMVC 的流程

控制反转,依赖注入(IOC):利用 java 的反射机制,系统启动时根据配置创建容器。

面向切面编程(AOP):利用 java 的动态代理,装饰接口中的方法,在实现业务的前后加入统一操作。

首先发送请求到核心控制器(DispatcherServlet),Spring 容器通过配置寻找业务控制器(Controller),使用适配器找到相应的业务,处理业务,封装数据,返回 ModelAndView 转发视图与数据,页面显示。

Spring AOP 的原理

面向切面的编程,允许程序模块化横向切割关注点,或横切典型的责任划分,如日志和事务管理。切面通过带有 @Aspect 注解的类实现。

使用了动态代理,装饰接口中的方法,在处理业务的前后加入统一操作。

Spring IOC 的原理

Spring IOC 负责创建对象,管理对象,通过依赖注入(DI),装配对象,配置对象,并且管理这些对象的整个生命周期。

IOC 依赖注入把应用的代码量降到最低,使应用容易测试,单元测试不再需要单例和 JNDI 查找机制,实现了松耦合。IOC容器支持加载服务时的饿汉式初始化和懒加载。

使用了反射机制,不直接 new 对象,以配置文件的方式,依赖注入,通过工厂模式,使用反射操纵类的属性、方法、构造器。

SpringMVC、Struts1、Struts2 的核心入口是什么?

SpringMVC 是 DispatcherServlet,Struts1 是 ActionServlet,Struts2 是 StrutsPrepareAndExecuteFilter。

SpringMVC 控制器是不是单例模式?

是单例模式,所以在多线程访问的时候有线程安全问题,要解决这个问题,不能用同步(影响性能),不在控制器里写字段就行了。

控制器的注解可以替换吗?

不可以,只能用 @Controller。

@RequestMapping 注解在类上,有什么用?

表示该类中所有响应请求的方法都以该地址作为父路径。

怎样把请求映射到方法上?怎样指定请求方式?

在方法上添加 @RequestMapping 注解。method = RequestMethod.POST | RequestMethod.GET。

怎样从拦截的方法里获得前台传入的参数?

方法声明相同名字的参数,也可以用一个对象参数接收,对象属性与传入的名字一样。

返回值如何处理?

根据配置文件中的 InternalResourceViewResolver,前缀 + 返回值 + 后缀。

返回对象 ModelMap,前台可用 el 表达式接收。

视图 + 数据:ModelAndView。

转发:forward、重定向:redirect。

返回对象给 Ajax:方法上加 @ResponseBody。

怎样实现 SpringMVC 拦截器?

<mvc:interceptors>
    <!-- 配置一个拦截器的Bean就可以了 默认是对所有请求都拦截 -->
    <bean id="myInterceptor" class="com.et.action.MyHandlerInterceptor"></bean>
    <!-- 只针对部分请求拦截 -->
    <mvc:interceptor>
       <mvc:mapping path="/modelMap.do" />
       <bean class="com.et.action.MyHandlerInterceptorAdapter" />
    </mvc:interceptor></mvc:interceptors>

BeanFactory 和 FactoryBean 的区别

BeanFactory 是工厂模式的一个实现,提供了控制反转功能,用来把应用的配置和依赖从正真的应用代码中分离。所有的 Bean 都是由 BeanFactory(IOC 容器)管理的。最常用的 BeanFactory 实现是 XmlBeanFactory 类,它根据 XML 文件中的定义加载 Bean。该容器从 XML 文件读取配置元数据,并用它去创建一个完全配置的系统或应用。

FactoryBean 是一个能生产或者修饰对象生成的工厂 Bean,它的实现与设计模式中的工厂模式和修饰器模式类似。

BeanFactory 和 ApplicationContext 的区别

BeanFactroy采用的是延迟加载形式来注入Bean的,只有在使用到某个 Bean 时,才进行加载实例化。启动时不能发现配置错误。

ApplicationContext 在容器启动时,一次性创建了所有的 Bean。启动时可以发现配置错误。

Spring 配置文件

是个 XML 文件,这个文件包含了类信息,描述了如何配置它们,以及如何相互调用。

ApplicationContext 的实现

FileSystemXmlApplicationContext:此容器从一个 XML 文件中加载 Bean 的定义,XML Bean 配置文件的全路径名必须提供给它的构造函数。

ClassPathXmlApplicationContext:此容器也从一个 XML 文件中加载 Bean 的定义,需要正确设置 classpath,因为这个容器将在 classpath 里找 Bean 配置。

WebXmlApplicationContext:此容器加载一个 XML 文件,此文件定义了所有 Bean。

有哪些不同类型的 IOC(依赖注入)方式?

构造器依赖注入:容器触发一个类的构造器来实现的,该类有一系列参数,每个参数代表一个对其它类的依赖。

Setter 方法注入:容器通过调用无参构造器,或无参 static 工厂方法,实例化 bean 之后,调用该 bean 的 setter 方法。

用构造器实现强制依赖,setter 方法实现可选依赖。

什么是 Spring Bean?

是 Spring 应用中的 java 对象,它们被 Spring IOC 容器初始化,装配,和管理。这些 Bean 通过容器中配置的元数据创建。Bean 中有个属性 scope,如果它被赋为 prototype,会每次产生一个新的对象,如果赋值为 singleton,就是单例对象。

一个 Spring Bean 定义包含什么?

容器必知的所有配置元数据,包括如何创建一个 Bean,它的生命周期详情及它的依赖。

如何给 Spring 容器提供配置元数据?

XML 配置文件。

基于注解的配置。

基于 Java 的配置。

Spring Bean 的生命周期,如何被管理的?

从配置中读取 Bean,并实例化,根据 Bean 的定义填充属性。

如果 Bean 实现了 BeanNameAware 接口,Spring 传递 Bean 的 ID 到 SetBeanName 方法。

如果 Bean 实现了 BeanFactoryAware 接口,Spring 传递 beanFactory 给 SetBeanFactory 方法。

如果有与 Bean 相关联的 BeanPostProcessors,Spring 会在 postProcesserBeforeInitialization() 中调用它们。

如果 Bean 实现了 IntializingBean,调用它的 afterPropertySet 方法,如果 Bean 声明了初始化方法,调用此初始化方法。

如果 Bean 实现了 DisposableBean,将调用 destroy()。

哪些是重要的 Bean 生命周期方法?能重载它们吗?

setup:在容器加载 Bean 的时候被调用。

teardown:在容器卸载类的时候被调用。

Bean 标签有两个重要的属性:init-method、destroy-method,用于定制初始化和注销方法。它们也有相应的注解(@PostConstruct、@PreDestroy)。

Spring Bean 的作用域

singleton: Bean 在每个 Spring IOC 容器中只有一个实例。

prototype:一个 Bean 的定义可以有多个实例。

request:每次 http 请求都会创建一个 Bean,该作用域仅在基于 web 的 Spring ApplicationContext 情形下有效。

session:在一个 HTTP Session 中,一个 Bean 定义对应一个实例。该作用域仅在基于 web 的 Spring ApplicationContext 情形下有效。

global-session:在一个全局的 HTTP Session 中,一个 Bean 定义对应一个实例。该作用域仅在基于 web 的 Spring ApplicationContext 情形下有效。

Spring Bean 的默认作用域是 Singleton。单例 Bean 不是线程安全的。

什么是 Spring 的内部 Bean?

当一个 Bean 仅被用作另一个 Bean 的属性时,可以被声明为一个内部 Bean,在 Spring 的基于 XML 的配置元数据中,可以在 property 或 constructor-arg 元素内使用 bean 元素,内部 Bean 通常是匿名的,它们的 Scope 一般是 prototype。

什么是 Bean 装配?

在 Spring 容器中把 Bean 组装到一起,前提是容器需要知道 Bean 的依赖关系。Spring 容器能够自动装配相互合作的 Bean。

自动装配的方式

no:默认的方式是不进行自动装配,通过显式设置 ref 属性来进行装配。

byName:通过参数名自动装配,Spring 容器在配置文件中发现 Bean 的 autowire 属性被设置成 byName,之后容器试图匹配、装配和该 Bean 的属性具有相同名字的 Bean。

byType:通过参数类型自动装配,Spring 容器在配置文件中发现 Bean 的 autowire 属性被设置成 byType,之后容器试图匹配、装配和该 Bean 的属性具有相同类型的 Bean。如果有多个 Bean 符合条件,则抛出错误。

constructor:这个方式类似于 byType,但是要提供给构造器参数,如果没有确定的带参数的构造器参数类型,将会抛出异常。

autodetect:首先尝试使用 constructor 来自动装配,如果无法工作,则使用 byType方式。

自动装配有哪些局限性?

重写: 仍需用配置来定义依赖,意味着总要重写自动装配。

基本数据类型:不能自动装配简单的属性,如基本数据类型,String 字符串,和类。

模糊特性:自动装配不如显式装配精确,如果有可能,建议使用显式装配。

可以在 Spring 中注入一个 null 或一个空字符串吗?

可以。

怎样开启注解装配?

注解装配在默认情况下是不开启的,为了使用注解装配,要在 Spring 配置文件中配置 <context:annotation-config/> 元素。

@Required 注解

这个注解表明 Bean 的属性必须在配置的时候设置,通过一个 Bean 定义的显式的属性值或通过自动装配,若 @Required 注解的 Bean 属性未被设置,容器将抛出 BeanInitializationException。

@Autowired 注解

提供了更细粒度的控制,包括在何处以及如何完成自动装配。它的用法和 @Required 一样,修饰 setter 方法、构造器、属性。

@Qualifier 注解

当有多个相同类型的 Bean 却只有一个需要自动装配时,将 @Qualifier 注解和 @Autowire 注解结合使用以消除这种混淆,指定需要装配的确切的 Bean。

在 Spring 框架中如何更有效地使用 JDBC?

使用 SpringJDBC 框架,资源管理和错误处理的代价都会被减轻。所以开发者只需写 statements 和 queries 从数据存取数据,JDBC 也可以在 Spring 框架提供的模板类的帮助下更有效地被使用,这个模板叫 JdbcTemplate。JdbcTemplate 类提供了很多便利的方法,例如,把数据库数据转变成基本数据类型或对象,执行写好的或可调用的数据库操作语句,提供自定义的数据错误处理。

Spring 对 DAO 的支持

简化 DAO 和数据访问技术如 JDBC,Hibernate or JDO 结合使用。可以方便切换持久层。编码时也不用担心会捕获每种技术特有的异常。

使用 Spring 通过什么方式访问 Hibernate?

控制反转:Hibernate Template 和 Callback。 继承:HibernateDAOSupport 提供一个 AOP 拦截器。

Spring 支持的事务管理类型

编程式事务管理:通过编程的方式管理事务,灵活性高,但是难维护。 声明式事务管理:可以将业务代码和事务管理分离,只需用注解和 XML 配置来管理事务。(推荐,对代码影响小)

Spring 框架的事务管理有哪些优点?

为不同的事务,如 JTA,JDBC,Hibernate,JPA 和JDO,提供一个不变的编程模式。

为编程式事务管理提供了一套简单的 API。

支持声明式事务管理。

与 Spring 各种数据访问抽象层很好得集成。

在 Spring AOP 中,关注点和横切关注的区别是什么?

关注点是应用中一个模块的行为,一个关注点可能会被定义成一个我们想实现的一个功能。

横切关注点是整个应用都会使用的功能,并影响整个应用,比如日志,安全和数据传输,几乎应用的每个模块都需要的功能。

连接点

连接点代表一个应用程序的某个位置,在这个位置我们可以插入一个 AOP 切面,它实际上是应用程序执行 Spring AOP 的位置。

通知

通知是个在方法执行前或执行后要做的动作,实际上是程序执行时要通过 Spring AOP 框架触发的代码段。

  • before:前置通知,在一个方法执行前被调用。

  • after: 在方法执行之后调用的通知,无论方法执行是否成功。

  • after-returning: 仅当方法成功完成后执行的通知。

  • after-throwing: 在方法抛出异常退出时执行的通知。

  • around: 在方法执行之前和之后调用的通知。

切点

切入点是一个或一组连接点,通知将在这些位置执行。可以通过表达式或匹配的方式指明切入点。

什么是引入?

引入允许我们在已存在的类中增加新的方法和属性。

什么是目标对象?

被一个或者多个切面所通知的对象。它通常是一个代理对象。也指被通知(advised)对象。

什么是代理?

代理是通知目标对象后创建的对象。从客户端的角度看,代理对象和目标对象是一样的。

什么是织入?

织入是将切面和到其他应用类型或对象连接或创建一个被通知对象的过程。织入可以在编译时,加载时,或运行时完成。

Spring 中用到了那些设计模式?

工厂模式 + 模板模式 + 单例模式:BeanFactory、ApplicationContext。

代理模式:AOP 用到了动态代理。

SpringMVC 的工作原理

用户发送请求至前端控制器DispatcherServlet。

DispatcherServlet 收到请求调用 HandlerMapping 处理器映射器。

处理器映射器找到具体的处理器(可以根据 xml 配置、注解进行查找),生成处理器对象及处理器拦截器,返回给DispatcherServlet。

DispatcherServlet 调用 HandlerAdapter 处理器适配器。

HandlerAdapter 经过适配调用具体的 Controller。

Controller 执行完成返回 ModelAndView。

HandlerAdapter 将 controller 执行结果 ModelAndView 返回给 DispatcherServlet。

DispatcherServlet 将 ModelAndView 传给 ViewReslover 视图解析器。

ViewReslover 解析后返回具体 View。

DispatcherServlet 根据 View 进行渲染视图。

DispatcherServlet 响应用户。

Spring 是如何管理事务的?

声明式事务管理使用了 AOP 面向切面编程实现的,本质就是在目标方法执行前后进行拦截。在目标方法执行前加入或创建一个事务,在执行方法执行后,根据实际情况选择提交或是回滚事务。

  • PlatformTransactionManager:事务管理器。主要用于平台相关事务的管理。

  • TransactionDefinition:事务定义信息。用来定义事务相关的属性,给事务管理器 PlatformTransactionManager 使用。

  • TransactionStatus:事务具体运行状态。事务管理过程中,每个时间点事务的状态信息。

Spring 的不同事务传播行为有哪些?分别是干什么用的?

Spring 循环注入的原理

就是两个或多个 Bean 相互之间的持有对方。

  • 构造器循环依赖:只能抛出BeanCurrentlyInCreationException异常表示循环依赖。

  • setter循环依赖:提前暴露刚完成构造器注入,但未完成其他步骤(如 setter 注入)的 Bean,只能解决单例作用域的Bean循环依赖。

如何保证 Controller 并发的安全?

在控制器中不使用实例变量。

将控制器的作用域从单例改为原型,在 Spring 配置文件 Controller 中声明 scope = "prototype",每次都创建新的 Controller。

在 Controller 中使用 ThreadLocal 变量。

发表评论

下一篇: 上一篇: