Spring和SpringBoot比较

2020-05-14 15:40发布

1、概述:

对于 Spring 和 SpringBoot 到底有什么区别,我听到了很多答案,刚开始迈入学习 SpringBoot 的我当时也是一头雾水,随着经验的积累、我慢慢理解了这两个框架到底有什么区别,我相信对于用了 SpringBoot 很久的开发人员来说,有绝大部分还不是很理解 SpringBoot 到底和 Spring 有什么区别,看完文章中的比较,或许你有了不同的答案和看法!
2、什么是Spring呢?

先来聊一聊 Spring 作为 Java 开发人员,大家都 Spring 可不陌生,简而言之, Spring 框架为开发 Java 应用程序提供了全面的基础架构支持。它包含一些很好的功能,如依赖注入和开箱即用的模块,如:

Spring JDBC 、Spring MVC 、Spring Security、 Spring AOP 、Spring ORM 、Spring Test

这些模块大家应该都用过吧,这些模块缩短应用程序的开发时间,提高了应用开发的效率

例如,在 Java Web 开发的早期阶段,我们需要编写大量的代码来将记录插入到数据源中。但是通过使用 Spring JDBC 模块的 JDBCTemplate ,我们可以将这操作简化为只需配置几行代码。
3、什么是Spring Boot呢?

Spring Boot 基本上是 Spring 框架的扩展,它消除了设置 Spring 应用程序所需的 XML配置,为更快,更高效的开发生态系统铺平了道路。

以下是 Spring Boot 中的一些特点:

1:创建独立的 spring 应用。

2:嵌入 Tomcat , Jetty Undertow 而且不需要部署他们。

3:提供的“starters” poms来简化 Maven 配置

4:尽可能自动配置 spring 应用。

5:提供生产指标,健壮检查和外部化配置

6:绝对没有代码生成和 XML 配置要求
4、让我们逐步熟悉这两个框架
4.1、 Maven依赖

首先,让我们看一下使用Spring创建Web应用程序所需的最小依赖项

<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-web</artifactId>
    <version>5.1.0.RELEASE</version>
</dependency>
<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-webmvc</artifactId>
    <version>5.1.0.RELEASE</version>
</dependency>

与Spring不同,Spring Boot只需要一个依赖项来启动和运行Web应用程序:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
    <version>2.0.6.RELEASE</version>
</dependency>

在进行构建期间,所有其他依赖项将自动添加到项目中。

另一个很好的例子就是测试库。我们通常使用 Spring Test , JUnit , Hamcrest 和 Mockito 库。在 Spring 项目中,我们应该将所有这些库添加为依赖项。但是在 Spring Boot中 ,我们只需要添加 spring-boot-starter-test 依赖项来自动包含这些库。

Spring Boot为不同的Spring模块提供了许多依赖项。一些最常用的是:

spring-boot-starter-data-jpa
spring-boot-starter-security
spring-boot-starter-test
spring-boot-starter-web
spring-boot-starter-thymeleaf
4.2、MVC配置

让我们来看一下 Spring 和 Spring Boot 创建 JSP Web 应用程序所需的配置。

Spring 需要定义调度程序 servlet ,映射和其他支持配置。我们可以使用 web.xml 文件或 Initializer 类来完成此操作:

public class MyWebAppInitializer implements WebApplicationInitializer {
   
    @Override
    public void onStartup(ServletContext container) {
        AnnotationConfigWebApplicationContext context = new AnnotationConfigWebApplicationContext();
        context.setConfigLocation("com.pingfangushi");
          container.addListener(new ContextLoaderListener(context));
          ServletRegistration.Dynamic dispatcher = container
          .addServlet("dispatcher", new DispatcherServlet(context));
        dispatcher.setLoadOnStartup(1);
        dispatcher.addMapping("/");
    }
}

还需要将 @EnableWebMvc 注释添加到 @Configuration 类,并定义一个视图解析器来解析从控制器返回的视图:

@EnableWebMvc
@Configuration
public class ClientWebConfig implements WebMvcConfigurer {  
   @Bean
   public ViewResolver viewResolver() {
      InternalResourceViewResolver bean
        = new InternalResourceViewResolver();
      bean.setViewClass(JstlView.class);
      bean.setPrefix("/WEB-INF/view/");
      bean.setSuffix(".jsp");
      return bean;
   }
}

和上述操作一比,一旦我们添加了 Web 启动程序, Spring Boot 只需要在 application 配置文件中配置几个属性来完成如上操作:

spring.mvc.view.prefix=/WEB-INF/jsp/
spring.mvc.view.suffix=.jsp

上面的所有Spring配置都是通过一个名为auto-configuration的过程添加 Boot web starter 来自动包含的。

这意味着 Spring Boot 将查看应用程序中存在的依赖项,属性和 bean ,并根据这些依赖项,对属性和 bean 进行配置。当然,如果我们想要添加自己的自定义配置,那么 Spring Boot 自动配置将会退回。
4.3、配置模板引擎

现在我们来看下如何在Spring和Spring Boot中配置Thymeleaf模板引擎。

在 Spring 中,我们需要为视图解析器添加 thymeleaf-spring5 依赖项和一些配置:

@Configuration
@EnableWebMvc
public class MvcWebConfig implements WebMvcConfigurer {
 
    @Autowired
    private ApplicationContext applicationContext;
 
    @Bean
    public SpringResourceTemplateResolver templateResolver() {
        SpringResourceTemplateResolver templateResolver = new SpringResourceTemplateResolver();
        templateResolver.setApplicationContext(applicationContext);
        templateResolver.setPrefix("/WEB-INF/views/");
        templateResolver.setSuffix(".html");
        return templateResolver;
    }
 
    @Bean
    public SpringTemplateEngine templateEngine() {
        SpringTemplateEngine templateEngine = new SpringTemplateEngine();
        templateEngine.setTemplateResolver(templateResolver());
        templateEngine.setEnableSpringELCompiler(true);
        return templateEngine;
    }
 
    @Override
    public void configureViewResolvers(ViewResolverRegistry registry) {
        ThymeleafViewResolver resolver = new ThymeleafViewResolver();
        resolver.setTemplateEngine(templateEngine());
        registry.viewResolver(resolver);
    }
}

SpringBoot1X 只需要 spring-boot-starter-thymeleaf 的依赖 项 来启用 Web 应用程序中的        Thymeleaf 支持。但是由于 Thymeleaf3.0 中的新功能, 我们必须将 thymeleaf-layout-dialect 添加 为 SpringBoot2X Web应用程序中的依赖项。一旦依赖关系到位,我们就可以将模板添加到 src/main/resources/templates 文件夹中, SpringBoot 将自动显示它们。
4.4、Spring Security 配置
为简单起见,我们使用框架默认的 HTTP Basic 身份验证。让我们首先看一下使用 Spring 启用Security 所需的依赖关系和配置。
Spring 首先需要依赖 spring-security-web 和 spring-security-config 模块。接下来, 我们需要添加一个扩展 WebSecurityConfigurerAdapter 的类,并使用 @EnableWebSecurity 注解:

@Configuration
@EnableWebSecurity
public class CustomWebSecurityConfigurerAdapter extends WebSecurityConfigurerAdapter {
   
    @Autowired
    public void configureGlobal(AuthenticationManagerBuilder auth) throws Exception {
        auth.inMemoryAuthentication()
          .withUser("admin")
            .password(passwordEncoder()
            .encode("password"))
          .authorities("ROLE_ADMIN");
    }
   
    @Override
    protected void configure(HttpSecurity http) throws Exception {
        http.authorizeRequests()
          .anyRequest().authenticated()
          .and()
          .httpBasic();
    }
    @Bean
    public PasswordEncoder passwordEncoder() {
        return new BCryptPasswordEncoder();
    }
}

这里我们使用 inMemoryAuthentication 来设置身份验证。同样, Spring Boot 也需要这些依赖项才能使其工作。但是我们只需要定义 spring-boot-starter-security 的依赖关系,因为这会自动将所有相关的依赖项添加到类路径中。

Spring Boot 中的安全配置与上面的相同。
5、应用程序引导配置

Spring 和 Spring Boot 中应用程序引导的基本区别在于 servlet 。

Spring 使用 web.xml 或 SpringServletContainerInitializer 作为其引导入口点。

Spring Boot 仅使用 Servlet 3 功能来引导应用程序,下面让我们详细来了解下
5.1、Spring 是怎样引导配置的呢?

Spring 支持传统的 web.xml 引导方式以及最新的 Servlet 3+ 方法。

让我们看一下 web.xml 方法的步骤:

Servlet 容器(服务器)读取 web.xml

web.xml 中定义的 DispatcherServlet 由容器实例化

DispatcherServlet 通过读取 WEB-INF / {servletName} -servlet.xml 来创建 WebApplicationContext

最后, DispatcherServlet 注册在应用程序上下文中定义的 bean

以下是使用 Servlet 3+ 方法的 Spring 引导:

容器搜索实现 ServletContainerInitializer 的类并执行

SpringServletContainerInitializer 找到实现所有类 WebApplicationInitializer

WebApplicationInitializer 创建具有XML或上下文 @Configuration 类

WebApplicationInitializer 创建 DispatcherServlet 的 与先前创建的上下文。
5.2、SpringBoot 有是如何配置的呢?

Spring Boot应用程序的入口点是使用@SpringBootApplication注释的类:

@SpringBootApplication
public class Application {
    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }
}

默认情况下, Spring Boot 使用嵌入式容器来运行应用程序。在这种情况下, Spring Boot 使用 public static void main 入口点来启动嵌入式 Web 服务器。此外,它还负责将 Servlet ,Filter 和 ServletContextInitializer bean 从应用程序上下文绑定到嵌入式 servlet 容器。

Spring Boot 的另一个特性是它会自动扫描同一个包中的所有类或 Main 类的子包中的组件。

Spring Boot 提供了将其部署到外部容器的方式。在这种情况下,我们必须扩展 SpringBootServletInitializer :

/**
* War部署
*
* @author SanLi
* Created by 2689170096@qq.com on 2018/4/15
*/
public class ServletInitializer extends SpringBootServletInitializer {
 
  @Override
  protected SpringApplicationBuilder configure(SpringApplicationBuilder application) {
      return application.sources(Application.class);
  }
 
  @Override
  public void onStartup(ServletContext servletContext) throws ServletException {
      super.onStartup(servletContext);
      servletContext.addListener(new HttpSessionEventPublisher());
  }
}

这里外部 servlet 容器查找在war包下的 META-INF 文件夹下MANIFEST.MF文件中定义的 Main-class , SpringBootServletInitializer 将负责绑定 Servlet , Filter 和 ServletContextInitializer 。
6、打包和部署

最后,让我们看看如何打包和部署应用程序。这两个框架都支持 Maven 和 Gradle 等通用包管理技术。但是在部署方面,这些框架差异很大。例如,Spring Boot Maven插件在 Maven 中提供Spring Boot 支持。它还允许打包可执行 jar 或 war 包并 就地 运行应用程序。

在部署环境中 Spring Boot 对比 Spring 的一些优点包括:

    提供嵌入式容器支持
    使用命令 java -jar 独立运行jar
    在外部容器中部署时,可以选择排除依赖关系以避免潜在的jar冲突
    部署时灵活指定配置文件的选项
    用于集成测试的随机端口生成

7、结论

简而言之,我们可以说 Spring Boot 只是 Spring 本身的扩展,使开发,测试和部署更加方便。