深度盘点一遍自动装配原理(上)

电子说

1.3w人已加入

描述

前言

Spring翻译为中文是“春天”,的确,在某段时间内,它给Java开发人员带来过春天,但是随着我们项目规模的扩大,Spring需要配置的地方就越来越多,夸张点说,“配置两小时,Coding五分钟”。这种纷繁复杂的xml配置随着软件行业一步步地发展,必将逐步退出历史舞台。


SpringBoot介绍

来自:百度百科

Spring Boot是由Pivotal团队提供的全新框架,其设计目的是用来简化新Spring应用的初始搭建以及开发过程。该框架使用了特定的方式来进行配置,从而使开发人员不再需要定义样板化的配置。通过这种方式,Spring Boot致力于在蓬勃发展的快速应用开发领域(rapid application development)成为领导者。

SpringBoot所具备的特征有:

  • 可以创建独立的Spring应用程序,并且基于其Maven或Gradle插件,可以创建可执行的JARs和WARs;
  • 内嵌Tomcat或Jetty等Servlet容器;
  • 提供自动配置的“starter”项目对象模型(POMS)以简化Maven配置;
  • 尽可能自动配置Spring容器;
  • 提供准备好的特性,如指标、健康检查和外部化配置;
  • 绝对没有代码生成,不需要XML配置。

自己的理解:

SpringBoot,顾名思义,给人的感觉就是让Spring启动的这么一个项目。在过去,我们要让一个Spring项目启动,往往需要配置很多的xml配置文件,但是在使用SpringBoot之后,我们甚至无需写一行xml,就可以直接将整个项目启动,这种“零配置”的做法减轻了开发人员很多的工作量,可以让开发人员一心扑在业务逻辑的设计上,使项目的逻辑更加完善。

除此之外,其采用了JavaConfig的配置风格,导入组件的方式也由原来的直接配置改为@EnableXXXX,这种纯Java代码的配置和导入组件的方式,使代码看上去更加的优雅,所以SpringBoot如今受到大小公司和大多数程序员的青睐,不是没有原因的。

SpringBoot之所以可以做到简化配置文件直接启动,无外乎是其内部的两种设计策略: 开箱即用和约定大于配置

开箱即用:在开发过程中,通过maven项目的pom文件中添加相关依赖包,然后通过相应的注解来代替繁琐的XML配置以管理对象的生命周期。

约定大于配置:由SpringBoot本身来配置目标结构,由开发者在结构中添加信息的软件设计范式。这一特点虽降低了部分灵活性,增加了BUG定位的复杂性,但减少了开发人员需要做出决定的数量,同时减少了大量的XML配置,并且可以将代码编译、测试和打包等工作自动化。

那么在这篇博客中,我们需要了解的所有东西,就应该从这两个特点出发,一步一步深入SpringBoot自动装配的原理。


开箱即用原理

要理解这一特点,首先要先自己体会开箱即用的整个过程带来的便利。

体验开箱即用

SpringBoot提供了我们快速创建SpringBoot项目的地方:https://start.spring.io/

我们只需要在这个网页中把整个项目起好名字,然后选好我们需要的组件,就可以直接获得一个可以跑起来的SpringBoot项目。

开发

我们只需要填完上述信息,点击Generate,就可以直接将一个SpringBoot项目下载下来,然后导入我们的IDE,Eclipse或者IDEA都可,之后就可以直接将它运行起来。

全项目结构:

开发

启动:

开发

访问:http://localhost:8080/

开发

代表整个SpringBoot项目启动成功。

开箱即用原理剖析

对比SSM配置

其实在上文的开箱即用中,我们相当于引入了一个SpringMVC的组件,但是大家可以看到,我们没有经过任何的配置就将项目启动了。反观过去SSM框架的SpringMVC配置,我这里有一份留存的大家可以对比一下。

spring-web.xml:

<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context"
    xmlns:mvc="http://www.springframework.org/schema/mvc"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
    http://www.springframework.org/schema/beans/spring-beans.xsd
    http://www.springframework.org/schema/context
    http://www.springframework.org/schema/context/spring-context.xsd
    http://www.springframework.org/schema/mvc
    http://www.springframework.org/schema/mvc/spring-mvc-3.2.xsd">
    
    
    
    <mvc:annotation-driven />

    
    <mvc:resources mapping="/resources/**" location="/resources/" />
    <mvc:default-servlet-handler />

    
    <bean id="viewResolver"
        class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <property name="prefix" value="/WEB-INF/html/"><span class="hljs-name"property>
        <property name="suffix" value=".html"><span class="hljs-name"property>
    <span class="hljs-name"bean>
    
    <bean id="multipartResolver"
        class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
        <property name="defaultEncoding" value="utf-8"><span class="hljs-name"property>
        <property name="maxUploadSize" value="10485760000"><span class="hljs-name"property>
        <property name="maxInMemorySize" value="20971520"><span class="hljs-name"property>
    <span class="hljs-name"bean>
    
    <bean
        class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter">
        <property name="messageConverters">
            <list>
                <bean
                    class="org.springframework.http.converter.StringHttpMessageConverter">
                    <property name="supportedMediaTypes">
                        <list>
                            <value>text/html;charset=UTF-8<span class="hljs-name"value>
                        <span class="hljs-name"list>
                    <span class="hljs-name"property>
                <span class="hljs-name"bean>
            <span class="hljs-name"list>
        <span class="hljs-name"property>
    <span class="hljs-name"bean>
    
    <context:component-scan base-package="com.SchoolShop.o2o.web" />
    
<span class="hljs-name"beans>

web.xml:


      name>spring-dispatcherclass="hljs-name"servlet-name>
      class>org.springframework.web.servlet.DispatcherServletclass="hljs-name"servlet-class>
      
          name>contextConfigLocationclass="hljs-name"param-name>
          value>classpath:spring/spring-*.xmlclass="hljs-name"param-value>
      class="hljs-name"init-param>
  class="hljs-name"servlet>
  mapping>
      name>spring-dispatcherclass="hljs-name"servlet-name>
      
      /class="hljs-name"url-pattern>
  class="hljs-name"servlet-mapping>

可以看到,这里需要配置两个文件,web.xml和spring-web.xml,配置可以说是相当繁重。

那么相对于这个,SpringBoot的开箱即用就显得特别方便,那么我们着重聊聊SpringBoot开箱即用的原理。

从pom.xml开始

SpringBoot的项目都会存在一个父依赖,按住Ctrl+鼠标左键,可以点进去。

<parent>
    <groupId>org.springframework.boot<span class="hljs-name"groupId>
    <artifactId>spring-boot-starter-parent<span class="hljs-name"artifactId>
    <version>2.2.1.RELEASE<span class="hljs-name"version>
    <relativePath/> 
<span class="hljs-name"parent>

点进去之后发现里面除了一些插件和配置文件的格式之外,还存在一个依赖。

<parent>
    <groupId>org.springframework.boot<span class="hljs-name"groupId>
    <artifactId>spring-boot-dependencies<span class="hljs-name"artifactId>
    <version>2.2.1.RELEASE<span class="hljs-name"version>
    <relativePath>../../spring-boot-dependencies<span class="hljs-name"relativePath>
<span class="hljs-name"parent>

于是再点进去,可以发现里面放了很多的依赖和依赖的版本号。由于这个文件实在太长了,所以这里只展示一部分。

开发

开发

所以我们可以得出第一个结论:

spring-boot-dependencies:作为父工程,存放了SpringBoot的核心依赖。我们在写或者引入一些SpringBoot依赖的时候,不需要指定版本,正是因为SpringBoot的父依赖已经帮我们维护了一套版本。

另外我们还可以看到,在父依赖中也帮我们写好了资源库,不用我们自己再去配置了。


      
        trueclass="hljs-name"filtering>
        ${basedir}/src/main/resourcesclass="hljs-name"directory>
        <includes>
            
          <include>**/application*.ymlclass="hljs-name"include>
          <include>**/application*.yamlclass="hljs-name"include>
          <include>**/application*.propertiesclass="hljs-name"include>
        class="hljs-name"includes>
      class="hljs-name"resource>
      
        ${basedir}/src/main/resourcesclass="hljs-name"directory>
        
          <exclude>**/application*.ymlclass="hljs-name"exclude>
          <exclude>**/application*.yamlclass="hljs-name"exclude>
          <exclude>**/application*.propertiesclass="hljs-name"exclude>
        class="hljs-name"excludes>
      class="hljs-name"resource>
class="hljs-name"resources>

启动器

<dependency>
    <groupId>org.springframework.boot<span class="hljs-name"groupId>
    <artifactId>spring-boot-starter<span class="hljs-name"artifactId>
    <version>2.2.1.RELEASE<span class="hljs-name"version>
<span class="hljs-name"dependency>

启动器就是SpringBoot的启动场景,比如我们要使用web相关的,那么就直接引入spring-boor-starter-web,那么他就会帮我们自动导入web环境下所有必需的依赖。

我们来看看启动器中存放了一些什么内容:

以spring-boot-starter为例:

<dependency>
      <groupId>org.springframework.boot<span class="hljs-name"groupId>
      <artifactId>spring-boot<span class="hljs-name"artifactId>
      <version>2.2.1.RELEASE<span class="hljs-name"version>
      <scope>compile<span class="hljs-name"scope>
    <span class="hljs-name"dependency>
    <dependency>
      <groupId>org.springframework.boot<span class="hljs-name"groupId>
      <artifactId>spring-boot-autoconfigure<span class="hljs-name"artifactId>
      <version>2.2.1.RELEASE<span class="hljs-name"version>
      <scope>compile<span class="hljs-name"scope>
    <span class="hljs-name"dependency>
    <dependency>
      <groupId>org.springframework.boot<span class="hljs-name"groupId>
      <artifactId>spring-boot-starter-logging<span class="hljs-name"artifactId>
      <version>2.2.1.RELEASE<span class="hljs-name"version>
      <scope>compile<span class="hljs-name"scope>
    <span class="hljs-name"dependency>
    <dependency>
      <groupId>jakarta.annotation<span class="hljs-name"groupId>
      <artifactId>jakarta.annotation-api<span class="hljs-name"artifactId>
      <version>1.3.5<span class="hljs-name"version>
      <scope>compile<span class="hljs-name"scope>
    <span class="hljs-name"dependency>
    <dependency>
      <groupId>org.springframework<span class="hljs-name"groupId>
      <artifactId>spring-core<span class="hljs-name"artifactId>
      <version>5.2.1.RELEASE<span class="hljs-name"version>
      <scope>compile<span class="hljs-name"scope>
    <span class="hljs-name"dependency>
    <dependency>
      <groupId>org.yaml<span class="hljs-name"groupId>
      <artifactId>snakeyaml<span class="hljs-name"artifactId>
      <version>1.25<span class="hljs-name"version>
      <scope>runtime<span class="hljs-name"scope>
    <span class="hljs-name"dependency>

其中存放了自动配置相关的依赖、日志相关依赖、还有Spring-core等依赖,这些依赖我们只需要导入一个spring-boor-starter就可以直接将其全部引入,而不需要再像以前那样逐个导入了。

SpringBoot会将所有的功能场景都封装成一个一个的启动器,供开发人员使用。

我们在使用的时候也可以直接去官网上找我们所需的启动器,直接将其引入。

获取启动器文档:

https://docs.spring.io/spring-boot/docs/2.2.1.RELEASE/reference/html/using-spring-boot.html#using-boot-starter

打开APP阅读更多精彩内容
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

全部0条评论

快来发表一下你的评论吧 !

×
20
完善资料,
赚取积分