• Spring面试题(六十道)


    1. 什么是spring?
      Spring 是个java企业级应用的开源开发框架。Spring主要用来开发Java应用,但是有些扩展是针对构建J2EE平台的web应用。Spring 框架目标是简化Java企业级应用开发,并通过POJO为基础的编程模型促进良好的编程习惯。

    2. 使用Spring框架的好处是什么?
      轻量:Spring 是轻量的,基本的版本大约2MB
      控制反转:Spring通过控制反转实现了松散耦合,对像们给出它们的依赖,而不是创建或查找依赖的对象们
      面向切面的编程(AOP):Spring支持面向切面的编程,并且把应用业务逻辑和系统服务分开
      容器:Spring 包含并管理应用中对象的生命周期和配置
      MVC框架:Spring的WEB框架是个精心设计的框架,是Web框架的一个很好的替代品
      事务管理:Spring 提供一个持续的事务管理接口,可以扩展到上至本地事务下至全局事务(JTA)
      异常处理:Spring 提供方便的API把具体技术相关的异常(比如由JDBC,Hibernate or JDO抛出的)转化为一致的unchecked 异常

    3、Spring Framework 中有多少个模块,它们分别是什么?
    Spring 核心容器 – 该层基本上是 Spring Framework 的核心。它包含以下模块:

    · Spring Core · Spring Bean · SpEL (Spring Expression Language)
    · Spring Context

    数据访问/集成 – 该层提供与数据库交互的支持。它包含以下模块:

    · JDBC (Java DataBase Connectivity)
    · ORM (Object Relational Mapping)
    · OXM (Object XML Mappers) · JMS (Java Messaging Service)
    · Transaction

    Web – 该层提供了创建 Web 应用程序的支持。它包含以下模块:

    · Web
    · Web – Servlet
    · Web – Socket
    · Web – Portlet

    AOP

    · 该层支持面向切面编程

    Instrumentation

    · 该层为类检测和类加载器实现提供支持。

    Test

    · 该层为使用 JUnit 和 TestNG 进行测试提供支持。

    几个杂项模块: Messaging – 该模块为 STOMP 提供支持。它还支持注解编程模型,该模型用于从WebSocket 客户端路由和处理 STOMP 消息。 Aspects – 该模块为与 AspectJ 的集成提供支持。

    4、什么是 Spring 配置文件?
    Spring 配置文件是 XML 文件。该文件主要包含类信息。它描述了这些类是如何配置以及相互引入的。但是,XML 配置文件冗长且更加干净。如果没有正确规划和编写,那么在大项目中管理变得非常困难。

    5、Spring 应用程序有哪些不同组件?
    Spring 应用一般有以下组件:

    · 接口 - 定义功能。
    · Bean 类 - 它包含属性,setter 和 getter 方法,函数等。
    · Spring 面向切面编程(AOP) - 提供面向切面编程的功能。
    · Bean 配置文件 - 包含类的信息以及如何配置它们。
    · 用户程序 - 它使用接口。

    6、使用 Spring 有哪些方式?
    使用 Spring 有以下方式:

    · 作为一个成熟的 Spring Web 应用程序。
    · 作为第三方 Web 框架,使用 Spring Frameworks 中间层。
    · 用于远程使用。
    · 作为企业级 Java Bean,它可以包装现有的 POJO(Plain Old JavaObjects)。

    7、什么是 Spring IOC 容器?
    Spring 框架的核心是 Spring 容器。容器创建对象,将它们装配在一起,配置它们并管理它们的完整生命周期。Spring 容器使用依赖注入来管理组成应用程序的组件。容器通过读取提供的配置元数据来接收对象进行实例化,配置和组装的指令。该元数据可以通过 XML,Java 注解或 Java 代码提供。

    8、什么是依赖注入?
    在依赖注入中,您不必创建对象,但必须描述如何创建它们。您不是直接在代码 中将组件和服务连接在一起,而是描述配置文件中哪些组件需要哪些服务。由 IoC 容器将它们装配在一起。

    9、可以通过多少种方式完成依赖注入?
    通常,依赖注入可以通过三种方式完成,即: · 构造函数注入 · setter 注入 · 接口注入 在 Spring Framework 中,仅使用构造函数和 setter 注入。

    10、Spring IoC 的实现机制。
    Spring 中的 IoC 的实现原理就是工厂模式加反射机制。 示例:

    interface Fruit {
    	public abstract void eat();
    }
    class Apple implements Fruit {
    	public void eat(){
    		System.out.println("Apple");
    	}
    }
    class Orange implements Fruit {
    	public void eat(){
    		System.out.println("Orange");
    	}
    }
    class Factory {
    	public static Fruit getInstance(String ClassName) {
    		Fruit f=null;
    		try {
    			f=(Fruit)Class.forName(ClassName).newInstance();
    		}
    		catch (Exception e) {
    			e.printStackTrace();
    		}
    		return f;
    	}
    }
    class Client {
    	public static void main(String[] a) {
    		Fruit f=Factory.getInstance("io.github.dunwu.spring.Apple");
    		if(f!=null){
    			f.eat();
    		}
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33

    11、spring 提供了哪些配置方式?
    基于 xml 配置 bean 所需的依赖项和服务在 XML 格式的配置文件中指定。这些配置文件通常包含许多 bean 定义和特定于应用程序的配置选项。它们通常以 bean 标签开头。例如:

    
    
    
    
    • 1
    • 2
    • 3

    基于注解配置 您可以通过在相关的类,方法或字段声明上使用注解,将 bean 配置为组件类本身,而不是使用 XML 来描述 bean 装配。默认情况下,Spring 容器中未打开注解装配。因此,您需要在使用它之前在 Spring 配置文件中启用它。例如:

    
    
    
    
    
    • 1
    • 2
    • 3
    • 4

    基于 Java API 配置
    Spring 的 Java 配置是通过使用 @Bean 和 @Configuration 来实现。 (1) @Bean 注解扮演与 元素相同的角色。 (2) @Configuration 类允许通过简单地调用同一个类中的其他 @Bean 方法来定义 bean 间依赖关系。 例如:

    @Configuration
    public class StudentConfig {
    	@Bean
    	public StudentBean myStudent() {
    		return new StudentBean();
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    12、spring bean 容器的生命周期是什么样的?
    spring bean 容器的生命周期流程如下:

    (1)Spring 容器根据配置中的 bean 定义中实例化 bean。
    (2)Spring 使用依赖注入填充所有属性,如 bean 中所定义的配置。
    (3)如果 bean 实现BeanNameAware 接口,则工厂通过传递 bean 的 ID 来调用setBeanName()。
    (4)如果 bean 实现 BeanFactoryAware 接口,工厂通过传递自身的实例来调用 setBeanFactory()。
    (5)如果存在与 bean 关联的任何BeanPostProcessors,则调用preProcessBeforeInitialization() 方法。
    (6)如果为 bean 指定了 init 方法( 的 init-method 属性),那么将调 用它。
    (7)最后,如果存在与 bean 关联的任何 BeanPostProcessors,则将调用 postProcessAfterInitialization() 方法。
    (8)如果 bean 实现DisposableBean 接口,当 spring 容器关闭时,会调用 destory()。
    (9)如果为bean 指定了 destroy 方法( 的 destroy-method 属性),那么将 调用它。

    13、什么是 spring 的内部 bean?
    只有将 bean 用作另一个 bean 的属性时,才能将 bean 声明为内部 bean。为了定义 bean,Spring 的基于 XML 的配置元数据在 或 中提供了 元素的使用。内部 bean 总是匿名的,它们总是作为原型。 例如,假设我们有一个 Student 类,其中引用了 Person 类。这里我们将只创建一个 Person 类实例并在 Student 中使用它。

    Student.java

    public class Student {
    	private Person person;
    	//Setters and Getters
    }
    public class Person {
    	private String name;
    	private String address;
    	//Setters and Getters
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    bean.xml

    
    
    
    
    
    
    
    
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

    14、什么是 spring 装配
    当 bean 在 Spring 容器中组合在一起时,它被称为装配或 bean 装配。Spring容器需要知道需要什么 bean 以及容器应该如何使用依赖注入来将 bean 绑定在一起,同时装配 bean。

    15、自动装配有哪些方式?
    Spring 容器能够自动装配 bean。也就是说,可以通过检查 BeanFactory 的内容让 Spring 自动解析 bean 的协作者。 自动装配的不同模式:

    no - 这是默认设置,表示没有自动装配。应使用显式 bean 引用进行装配。
    byName - 它根据 bean 的名称注入对象依赖项。它匹配并装配其属性与 XML文件中由相同名称定义的 bean。
    byType - 它根据类型注入对象依赖项。如果属性的类型与 XML 文件中的一个 bean 名称匹配,则匹配并装配属性。构造函数- 它通过调用类的构造函数来注入依赖项。它有大量的参数。
    autodetect - 首先容器尝试通过构造函数使用 autowire 装配,如果不能,则尝试通过 byType 自动装配。

    16、自动装配有什么局限?
    覆盖的可能性 - 您始终可以使用 和 设置指定依赖项,这将覆盖自动装配。基本元数据类型 - 简单属性(如原数据类型,字符串和类)无法自动装配。令人困惑的性质 - 总是喜欢使用明确的装配,因为自动装配不太精确。

    17、pring由哪些模块组成?
    以下是Spring 框架的基本模块:
    1、Core module
    2、Bean module
    3、Context module
    4、Expression Language module
    5、JDBC module
    6、ORM module
    7、OXM module
    8、Java Messaging Service(JMS) module
    9、Transaction module
    10、Web module
    11、Web-Servlet module
    12、eb-Struts module
    13、Web-Portlet module

    18、Bean 工厂和 Application contexts 有什么区别?
    Application contexts提供一种方法处理文本消息,一个通常的做法是加载文件资源(比如镜像),它们可以向注册为监听器的bean发布事件。另外,在容器或容器内的对象上执行的那些不得不由bean工厂以程序化方式处理的操作,可以在Application contexts中以声明的方式处理。Application contexts实现了MessageSource接口,该接口的实现以可插拔的方式提供获取本地化消息的方法。

    19、谈谈自己对于Spring IOC和AOP的理解

    IOC
    IOC(Inversion Of Controll,控制反转)是一种设计思想,就是将原本在程序中手动创建对象的控制权,交由给Spring框架来管理。IOC在其他语言中也有应用,并非Spring特有。IOC容器是Spring用来实现IOC的载体,IOC容器实际上就是一个Map(key, value),Map中存放的是各种对象。
    将对象之间的相互依赖关系交给IOC容器来管理,并由IOC容器完成对象的注入。这样可以很大程度上简化应用的开发,把应用从复杂的依赖关系中解放出来。IOC容器就像是一个工厂一样,当我们需要创建一个对象的时候,只需要配置好配置文件/注解即可,完全不用考虑对象是如何被创建出来的。在实际项目中一个Service类可能由几百甚至上千个类作为它的底层,假如我们需要实例化这个Service,可能要每次都搞清楚这个Service所有底层类的构造函数,这可能会把人逼疯。如果利用IOC的话,你只需要配置好,然后在需要的地方引用就行了,大大增加了项目的可维护性且降低了开发难度。
    Spring时代我们一般通过XML文件来配置Bean,后来开发人员觉得用XML文件来配置不太好,于是Sprng Boot注解配置就慢慢开始流行起来。
    AOP
    AOP(Aspect-Oriented Programming,面向切面编程)能够将那些与业务无关,却为业务模块所共同调用的逻辑或责任(例如事务处理、日志管理、权限控制等)封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可扩展性和可维护性。
    Spring AOP是基于动态代理的,如果要代理的对象实现了某个接口,那么Spring AOP就会使用JDK动态代理去创建代理对象;而对于没有实现接口的对象,就无法使用JDK动态代理,转而使用CGlib动态代理生成一个被代理对象的子类来作为代理。
    当然也可以使用AspectJ,Spring AOP中已经集成了AspectJ,AspectJ应该算得上是Java生态系统中最完整的AOP框架了。使用AOP之后我们可以把一些通用功能抽象出来,在需要用到的地方直接使用即可,这样可以大大简化代码量。我们需要增加新功能也方便,提高了系统的扩展性。日志功能、事务管理和权限管理等场景都用到了AOP。

    20、Spring AOP和AspectJ AOP有什么区别?
    Spring AOP是属于运行时增强,而AspectJ是编译时增强。Spring AOP基于代理(Proxying),而AspectJ基于字节码操作(Bytecode Manipulation)。
    Spring AOP已经集成了AspectJ,AspectJ应该算得上是Java生态系统中最完整的AOP框架了。AspectJ相比于Spring AOP功能更加强大,但是Spring AOP相对来说更简单。
    如果我们的切面比较少,那么两者性能差异不大。但是,当切面太多的话,最好选择AspectJ,它比SpringAOP快很多。

    21、Spring中的bean的作用域有哪些?
    1.singleton:唯一bean实例,Spring中的bean默认都是单例的。
    2.prototype:每次请求都会创建一个新的bean实例。
    3.request:每一次HTTP请求都会产生一个新的bean,该bean仅在当前HTTP request内有效。
    4.session:每一次HTTP请求都会产生一个新的bean,该bean仅在当前HTTP session内有效。
    5.global-session:全局session作用域,仅仅在基于Portlet的Web应用中才有意义,Spring5中已经没有了。Portlet是能够生成语义代码(例如HTML)片段的小型Java Web插件。它们基于Portlet容器,可以像Servlet一样处理HTTP请求。但是与Servlet不同,每个Portlet都有不同的会话。

    22、Spring中的单例bean的线程安全问题了解吗?
    大部分时候我们并没有在系统中使用多线程,所以很少有人会关注这个问题。单例bean存在线程问题,主要是因为当多个线程操作同一个对象的时候,对这个对象的非静态成员变量的写操作会存在线程安全问题。
    有两种常见的解决方案:
    1.在bean对象中尽量避免定义可变的成员变量(不太现实)。
    2.在类中定义一个ThreadLocal成员变量,将需要的可变成员变量保存在ThreadLocal中(推荐的一种方式)。

    23、Spring中的bean生命周期?
    1.Bean容器找到配置文件中Spring Bean的定义。
    2.Bean容器利用Java Reflection API创建一个Bean的实例。
    3.如果涉及到一些属性值,利用set()方法设置一些属性值。
    4.如果Bean实现了BeanNameAware接口,调用setBeanName()方法,传入Bean的名字。
    5.如果Bean实现了BeanClassLoaderAware接口,调用setBeanClassLoader()方法,传入ClassLoader对象的实例。
    6.如果Bean实现了BeanFactoryAware接口,调用setBeanClassFacotory()方法,传入ClassLoader对象的实例。
    7.与上面的类似,如果实现了其他*Aware接口,就调用相应的方法。
    8.如果有和加载这个Bean的Spring容器相关的BeanPostProcessor对象,执行postProcessBeforeInitialization()方法。
    9.如果Bean实现了InitializingBean接口,执行afeterPropertiesSet()方法。
    10.如果Bean在配置文件中的定义包含init-method属性,执行指定的方法。
    11.如果有和加载这个Bean的Spring容器相关的BeanPostProcess对象,执行postProcessAfterInitialization()方法。
    12.当要销毁Bean的时候,如果Bean实现了DisposableBean接口,执行destroy()方法。
    13.当要销毁Bean的时候,如果Bean在配置文件中的定义包含destroy-method属性,执行指定的方法。

    24、Spring框架中用到了哪些设计模式
    1.工厂设计模式:Spring使用工厂模式通过BeanFactory和ApplicationContext创建bean对象。
    2.代理设计模式:Spring AOP功能的实现。
    3.单例设计模式:Spring中的bean默认都是单例的。
    4.模板方法模式:Spring中的jdbcTemplate、hibernateTemplate等以Template结尾的对数据库操作的类,它们就使用到了模板模式。
    5.包装器设计模式:我们的项目需要连接多个数据库,而且不同的客户在每次访问中根据需要会去访问不同的数据库。这种模式让我们可以根据客户的需求能够动态切换不同的数据源。
    6.观察者模式:Spring事件驱动模型就是观察者模式很经典的一个应用。
    7.适配器模式:Spring AOP的增强或通知(Advice)使用到了适配器模式、Spring MVC中也是用到了适配器模式适配Controller。

    25、@Component和@Bean的区别是什么
    1.作用对象不同。@Component注解作用于类,而@Bean注解作用于方法。
    2.@Component注解通常是通过类路径扫描来自动侦测以及自动装配到Spring容器中(我们可以使用@ComponentScan注解定义要扫描的路径)。@Bean注解通常是在标有该注解的方法中定义产生这个bean,告诉Spring这是某个类的实例,当我需要用它的时候还给我。
    3.@Bean注解比@Component注解的自定义性更强,而且很多地方只能通过@Bean注解来注册bean。比如当引用第三方库的类需要装配到Spring容器的时候,就只能通过@Bean注解来实现。
    @Bean注解的使用示例:

    @Configuration
    public class AppConfig {
        @Bean
        public TransferService transferService() {
            return new TransferServiceImpl();
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    上面的代码相当于下面的XML配置:

    
        
    
    
    • 1
    • 2
    • 3

    下面这个例子是无法通过@Component注解实现的:

    @Bean
    public OneService getService(status) {
        case (status)  {
            when 1:
                    return new serviceImpl1();
            when 2:
                    return new serviceImpl2();
            when 3:
                    return new serviceImpl3();
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    26、将一个类声明为Spring的bean的注解有哪些?
    我们一般使用@Autowired注解去自动装配bean。而想要把一个类标识为可以用@Autowired注解自动装配的bean,可以采用以下的注解实现:
    1.@Component注解。通用的注解,可标注任意类为Spring组件。如果一个Bean不知道属于哪一个层,可以使用@Component注解标注。
    2.@Repository注解。对应持久层,即Dao层,主要用于数据库相关操作。
    3.@Service注解。对应服务层,即Service层,主要涉及一些复杂的逻辑,需要用到Dao层(注入)。
    4.@Controller注解。对应Spring MVC的控制层,即Controller层,主要用于接受用户请求并调用Service层的方法返回数据给前端页面。

    27、Spring事务管理的方式有几种?
    1.编程式事务:在代码中硬编码(不推荐使用)。
    2.声明式事务:在配置文件中配置(推荐使用),分为基于XML的声明式事务和基于注解的声明式事务。

    28、Spring事务中的隔离级别有哪几种?
    在TransactionDefinition接口中定义了五个表示隔离级别的常量:
    ISOLATION_DEFAULT:使用后端数据库默认的隔离级别,Mysql默认采用的REPEATABLE_READ隔离级别;Oracle默认采用的READ_COMMITTED隔离级别。
    ISOLATION_READ_UNCOMMITTED:最低的隔离级别,允许读取尚未提交的数据变更,可能会导致脏读、幻读或不可重复读。
    ISOLATION_READ_COMMITTED:允许读取并发事务已经提交的数据,可以阻止脏读,但是幻读或不可重复读仍有可能发生
    ISOLATION_REPEATABLE_READ:对同一字段的多次读取结果都是一致的,除非数据是被本身事务自己所修改,可以阻止脏读和不可重复读,但幻读仍有可能发生。
    ISOLATION_SERIALIZABLE:最高的隔离级别,完全服从ACID的隔离级别。所有的事务依次逐个执行,这样事务之间就完全不可能产生干扰,也就是说,该级别可以防止脏读、不可重复读以及幻读。但是这将严重影响程序的性能。通常情况下也不会用到该级别。

    29、Spring事务中有哪几种事务传播行为?
    在TransactionDefinition接口中定义了八个表示事务传播行为的常量。
    支持当前事务的情况:
    PROPAGATION_REQUIRED:如果当前存在事务,则加入该事务;如果当前没有事务,则创建一个新的事务。
    PROPAGATION_SUPPORTS: 如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。
    PROPAGATION_MANDATORY: 如果当前存在事务,则加入该事务;如果当前没有事务,则抛出异常。(mandatory:强制性)。
    不支持当前事务的情况:
    PROPAGATION_REQUIRES_NEW: 创建一个新的事务,如果当前存在事务,则把当前事务挂起。
    PROPAGATION_NOT_SUPPORTED: 以非事务方式运行,如果当前存在事务,则把当前事务挂起。
    PROPAGATION_NEVER: 以非事务方式运行,如果当前存在事务,则抛出异常。
    其他情况:
    PROPAGATION_NESTED: 如果当前存在事务,则创建一个事务作为当前事务的嵌套事务来运行;如果当前没有事务,则该取值等价于PROPAGATION_REQUIRED。

    30、什么是Spring beans?
    Spring beans 是那些形成Spring应用的主干的java对象。它们被Spring IOC容器初始化,装配,和管理。这些beans通过容器中配置的元数据创建。比如,以XML文件中 的形式定义。
    Spring 框架定义的beans都是单件beans。在bean tag中有个属性”singleton”,如果它被赋为TRUE,bean 就是单件,否则就是一个 prototype bean。默认是TRUE,所以所有在Spring框架中的beans 缺省都是单件。

    31、什么是Spring的内部bean?
    当一个bean仅被用作另一个bean的属性时,它能被声明为一个内部bean,为了定义inner bean,在Spring 的 基于XML的 配置元数据中,可以在 或 元素内使用 元素,内部bean通常是匿名的,它们的Scope一般是prototype。

    32、什么是bean装配?
    装配,或bean 装配是指在Spring 容器中把bean组装到一起,前提是容器需要知道bean的依赖关系,如何通过依赖注入来把它们装配到一起。

    1. 什么是bean的自动装配?
      Spring 容器能够自动装配相互合作的bean,这意味着容器不需要和配置,能通过Bean工厂自动处理bean之间的协作。

    2. 解释不同方式的自动装配
      有五种自动装配的方式,可以用来指导Spring容器用自动装配方式来进行依赖注入
      no:默认的方式是不进行自动装配,通过显式设置ref 属性来进行装配。
      byName:通过参数名 自动装配,Spring容器在配置文件中发现bean的autowire属性被设置成byname,之后容器试图匹配、装配和该bean的属性具有相同名字的bean。
      byType:通过参数类型自动装配,Spring容器在配置文件中发现bean的autowire属性被设置成byType,之后容器试图匹配、装配和该bean的属性具有相同类型的bean。如果有多个bean符合条件,则抛出错误。
      constructor:这个方式类似于byType, 但是要提供给构造器参数,如果没有确定的带参数的构造器参数类型,将会抛出异常。
      autodetect:首先尝试使用constructor来自动装配,如果无法工作,则使用byType方式。

    35.自动装配有哪些局限性?
    自动装配的局限性是:
    重写:你仍需用 和 配置来定义依赖,意味着总要重写自动装配。
    基本数据类型:你不能自动装配简单的属性,如基本数据类型,String字符串,和类。
    模糊特性:自动装配不如显式装配精确,如果有可能,建议使用显式装配。

    36、什么是 Spring IOC 容器?
    Spring 框架的核心是 Spring 容器。容器创建对象,将它们装配在一起,配置它们并管理它们的完整生命周期。Spring 容器使用依赖注入来管理组成应用程序的组件。
    容器通过读取提供的配置元数据来接收对象进行实例化,配置和组装的指令。该元数据可以通过 XML,Java 注解或 Java 代码提供。

    37、什么是依赖注入?
    在依赖注入中,您不必创建对象,但必须描述如何创建它们。您不是直接在代码中将组件和服务连接在一起,而是描述配置文件中哪些组件需要哪些服务。由 IoC 容器将它们装配在一起。

    38、可以通过多少种方式完成依赖注入?
    通常,依赖注入可以通过三种方式完成,即:
    构造函数注入
    setter 注入
    接口注入
    在 Spring Framework 中,仅使用构造函数和 setter 注入。

    39、区分构造函数注入和 setter 注入。
    构造函数注入 setter 注入
    没有部分注入 有部分注入
    构造函数注入 setter 注入
    不会覆盖 setter 属性 会覆盖 setter 属性
    任意修改都会创建一个新实例 任意修改不会创建一个新实例
    适用于设置很多属性 适用于设置少量属性

    40、spring 中有多少种 IOC 容器?

    • BeanFactory - BeanFactory 就像一个包含 bean 集合的工厂类。它会在客户端要求时实例化 bean。
    • ApplicationContext - ApplicationContext 接口扩展了 BeanFactory 接口。它在
      BeanFactory 基础上提供了一些额外的功能。

    41、区分 BeanFactory 和 ApplicationContext。
    BeanFactory ApplicationContext
    它使用懒加载 它使用即时加载
    它使用语法显式提供资源对象 它自己创建和管理资源对象
    不支持国际化 支持国际化
    不支持基于依赖的注解 支持基于依赖的注解

    42、列举 IoC 的一些好处。
    IoC 的一些好处是:
    它将最小化应用程序中的代码量。
    它将使您的应用程序易于测试,因为它不需要单元测试用例中的任何单例或 JNDI 查找机制。
    它以最小的影响和最少的侵入机制促进松耦合。
    它支持即时的实例化和延迟加载服务。

    43、Spring IoC 的实现机制。
    Spring 中的 IoC 的实现原理就是工厂模式加反射机制。
    示例:

    interface Fruit {
         public abstract void eat();
    }
    class Apple implements Fruit {
        public void eat(){
            System.out.println("Apple");
        }
    }
    class Orange implements Fruit {
        public void eat(){
            System.out.println("Orange");
        }
    }
    class Factory {
        public static Fruit getInstance(String ClassName) {
            Fruit f=null;
            try {
                f=(Fruit)Class.forName(ClassName).newInstance();
            } catch (Exception e) {
                e.printStackTrace();
            }
            return f;
        }
    }
    class Client {
        public static void main(String[] a) {
            Fruit f=Factory.getInstance("io.github.dunwu.spring.Apple");
            if(f!=null){
                f.eat();
            }
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32

    44、spring 支持集中 bean scope?
    Spring bean 支持 5 种 scope:

    • Singleton - 每个 Spring IoC 容器仅有一个单实例。
    • Prototype - 每次请求都会产生一个新的实例。
    • Request - 每一次 HTTP 请求都会产生一个新的实例,并且该 bean 仅在当前 HTTP 请求内有效。
    • Session - 每一次 HTTP 请求都会产生一个新的 bean,同时该 bean 仅在当前 HTTP session 内有效。
    • Global-session - 类似于标准的 HTTP Session 作用域,不过它仅仅在基于 portlet 的 web 应用中才有意义。Portlet 规范定义了全局 Session 的概念,它被所有构成某个 portlet web 应用的各种不同的 portlet 所共享。在 global session 作用域中定义的 bean 被限定于全局 portlet Session 的生命周期范围内。如果你在 web 中使用 global session 作用域来标识 bean,那么 web 会自动当成 session 类型来使用。
      仅当用户使用支持 Web 的 ApplicationContext 时,最后三个才可用。

    45、你用过哪些重要的 Spring 注解?
    @Controller - 用于 Spring MVC 项目中的控制器类。
    @Service - 用于服务类。
    @RequestMapping - 用于在控制器处理程序方法中配置 URI 映射。
    @ResponseBody - 用于发送 Object 作为响应,通常用于发送 XML 或 JSON 数据作为响应。
    @PathVariable - 用于将动态值从 URI 映射到处理程序方法参数。
    @Autowired - 用于在 spring bean 中自动装配依赖项。
    @Qualifier - 使用 @Autowired 注解,以避免在存在多个 bean 类型实例时出现混淆。
    @Scope - 用于配置 spring bean 的范围。
    @Configuration,@ComponentScan 和 @Bean - 用于基于 java 的配置。
    @Aspect,@Before,@After,@Around,@Pointcut - 用于切面编程(AOP)。

    46、如何在 spring 中启动注解装配?
    默认情况下,Spring 容器中未打开注解装配。因此,要使用基于注解装配,我们必须通过配置 元素在 Spring 配置文件中启用它。

    1. @Component, @Controller, @Repository, @Service 有何区别?
      @Component:这将 java 类标记为 bean。它是任何 Spring 管理组件的通用构造型。spring
      的组件扫描机制现在可以将其拾取并将其拉入应用程序环境中。
      @Controller:这将一个类标记为 Spring Web MVC 控制器。标有它的 Bean 会自动导入到 IoC 容器中。
      @Service:此注解是组件注解的特化。它不会对 @Component 注解提供任何其他行为。您可以在服务层类中使用 @Service
      而不是 @Component,因为它以更好的方式指定了意图。
      @Repository:这个注解是具有类似用途和功能的 @Component 注解的特化。它为 DAO 提供了额外的好处。它将 DAO
      导入 IoC 容器,并使未经检查的异常有资格转换为 Spring DataAccessException。

    48、@Required 注解有什么用?
    @Required 应用于 bean 属性 setter 方法。此注解仅指示必须在配置时使用 bean 定义中的显式属性值或使用自动装配填充受影响的 bean 属性。如果尚未填充受影响的 bean 属性,则容器将抛出 BeanInitializationException。
    示例:

    public class Employee {
        private String name;
        @Required
        public void setName(String name){
            this.name=name;
        }
    
    
        public string getName(){
            return name;
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    49、@Autowired 注解有什么用?
    @Autowired 可以更准确地控制应该在何处以及如何进行自动装配。此注解用于在 setter 方法,构造函数,具有任意名称或多个参数的属性或方法上自动装配 bean。默认情况下,它是类型驱动的注入。

    public class Employee {
        private String name;
        @Autowired
        public void setName(String name) {
            this.name=name;
        }
        public string getName(){
            return name;
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

    50、@Qualifier 注解有什么用?
    当您创建多个相同类型的 bean 并希望仅使用属性装配其中一个 bean 时,您可以使用@Qualifier 注解和 @Autowired 通过指定应该装配哪个确切的 bean 来消除歧义。
    例如,这里我们分别有两个类,Employee 和 EmpAccount。在 EmpAccount 中,使用@Qualifier 指定了必须装配 id 为 emp1 的 bean。

    public class Employee {
        private String name;
        @Autowired
        public void setName(String name) {
            this.name=name;
        }
        public string getName() {
            return name;
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

    EmpAccount.java

    public class EmpAccount {
        private Employee emp;
    
        @Autowired
        @Qualifier(emp1)
        public void showName() {
            System.out.println(“Employee name : ”+emp.getName);
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    51、@RequestMapping 注解有什么用?
    @RequestMapping 注解用于将特定 HTTP 请求方法映射到将处理相应请求的控制器中的特定类/方法。此注解可应用于两个级别:
    类级别:映射请求的 URL
    方法级别:映射 URL 以及 HTTP 请求方法

    52、spring DAO 有什么用?
    Spring DAO 使得 JDBC,Hibernate 或 JDO 这样的数据访问技术更容易以一种统一的方式工作。这使得用户容易在持久性技术之间切换。它还允许您在编写代码时,无需考虑捕获每种技术不同的异常。

    53、spring JDBC API 中存在哪些类?
    JdbcTemplate
    SimpleJdbcTemplate
    NamedParameterJdbcTemplate
    SimpleJdbcInsert
    SimpleJdbcCall

    54、什么是通知(Advice)?
    特定 JoinPoint 处的 Aspect 所采取的动作称为 Advice。Spring AOP 使用一个 Advice 作为拦截器,在 JoinPoint “周围”维护一系列的拦截器。

    1. 有哪些类型的通知(Advice)?
      Before - 这些类型的 Advice 在 joinpoint 方法之前执行,并使用 @Before 注解标记进行配置。
      After Returning - 这些类型的 Advice 在连接点方法正常执行后执行,并使用@AfterReturning
      注解标记进行配置。
      After Throwing - 这些类型的 Advice 仅在 joinpoint 方法通过抛出异常退出并使用@AfterThrowing 注解标记配置时执行。
      After (finally) - 这些类型的 Advice 在连接点方法之后执行,无论方法退出是正常还是异常返回,并使用 @After注解标记进行配置。
      Around - 这些类型的 Advice 在连接点之前和之后执行,并使用 @Around 注解标记进行配置。

    2. 指出在 spring aop 中 concern 和 cross-cutting concern 的不同之处。
      concern 是我们想要在应用程序的特定模块中定义的行为。它可以定义为我们想要实现的功能。
      cross-cutting concern 是一个适用于整个应用的行为,这会影响整个应用程序。例如,日志记录,安全性和数据传输是应用程序几乎每个模块都需要关注的问题,因此它们是跨领域的问题。

    3. AOP 有哪些实现方式?
      实现 AOP 的技术,主要分为两大类:
      静态代理 - 指使用 AOP 框架提供的命令进行编译,从而在编译阶段就可生成 AOP 代理类,因此也称为编译时增强;
      编译时编织(特殊编译器实现)
      类加载时编织(特殊的类加载器实现)。
      动态代理 - 在运行时在内存中“临时”生成 AOP 动态代理类,因此也被称为运行时增强。
      JDK 动态代理

    58、如何理解 Spring 中的代理?
    将 Advice 应用于目标对象后创建的对象称为代理。在客户端对象的情况下,目标对象和代理对象是相同的。

    Advice + Target Object = Proxy

    59、@SpringBootApplication的作用是什么?
    @SpringBootApplication注解是在springboot1.2.0中引入的,它支持自动配置特性。

    此批注封装了三种不同注解的作用:

    @Configuration:允许开发人员显式地注册bean
    @ComponentScan:启用组件扫描,这样控制器类和其他组件将被自动发现并注册为Spring应用程序上下文中的bean
    @EnableAutoConfiguration:启动SpringBoot的自动配置特性
    此注解可以使用以下可选参数:

    exclude:从自动配置中排除的类列表
    excludeNames:从自动配置中排除完全限定类名的列表
    scanBasePackage:其中提供了扫描程序包的列表
    scanBasePackageClasses:提供必须应用于扫描的其他包中的类的列表

    60、AOP有哪些优点?
    a.它是非侵入性的
    你的service/domain类被切面通知,而无需在service/domain类中添加任何与spring aop相关的类或接口。
    允许开发人员专注于业务代码,而不是切点。
    b.它是用纯Java实现的
    不需要特殊的编译单元或特殊的类加载器
    c.它使用Spring的IOC进行依赖注入
    切面可以配置为普通的spring bean。
    d.与任何其他AOP框架一样,它将横切关注点编织到类中,而不从这些类中调用横切关注点。
    e.集中或模块化横切关注点
    易于维护和更改切面
    改变需要在一处进行。
    在你不希望有的类中有日志记录,可以通过修改相应切面(日志切面)中的切入点来轻松实现。所以你只需要在一个地方做出改变。
    f.用于使用基于模式(XML配置)或@AspectJ基于注解的方式创建切面。
    g.易于配置

  • 相关阅读:
    go语言工具
    Mrtrix3---FACT--确定性纤维束追踪
    云栖探馆!云小宝首秀遇上老司机小龙,猜猜谁赢了?
    使用pytorch利用神经网络原理进行图片的训练(持续学习中....)
    Tomcat 架构
    在WSL中基于命令行打开Windows资源管理器
    Redis数据缓存(Redis的缓存击穿和穿透的区别)
    创新共享经济:探索Web3对新商业模式的启迪
    代码重构常用的技巧
    小米蓝牙耳机怎么选?适合小米手机的蓝牙耳机推荐
  • 原文地址:https://blog.csdn.net/mingshengda/article/details/126214387