• (转)Spring Boot(一)


    (二期)4、springboot的综合讲解

    【课程四】springbo...概念.xmind64.5KB

    【课程四】spring装配方式.xmind0.2MB

    【课程四预习】spri...解读.xmind0.1MB

    【课程四】springbo...过程.xmind0.3MB

    git demo : https://gitee.com/lv-success/git-second/tree/master/course-4-springboot/spring-boot-deep

    讲课顺序:

    • 第一节课程
    • springboot与springmvc的关系
    • springboot与微服务
    • springboot的自动装配
    • 认识springApplication
    • springboot的启动过程
    • 第二节课程
    • springboot的异常处理
    • springboot集成redis
    • springboot集成spring session
    • lombok的使用
    • 作业布置
    什么是spring boot

    其设计目的是用来简化新Spring应用的初始搭建以及开发过程。该框架使用了特定的方式来进行配置,从而使开发人员不再需要定义样板化的配置。

    以前在写spring项目的时候,要配置各种xml文件。特别是做这种框架集成,比如说ssm框架,就需要配置一堆的xml,还经常会出问题,这样就降低了开发效率。

    随着spring3,spring4的相继推出,约定大于配置逐渐成为了开发者的共识,大家也渐渐的从写xml转为各种注解的形式,在spring4的项目里,你甚至可以一行xml都不写。

    但虽然spring4已经可以做到无xml,写一个大项目需要导入很多的第三方包,maven配置要写几百行,这是一件很可怕的事。为了简化我们的配置,快速集成常用的第三方框架,就有了springboot的诞生。

    那么,spring boot可以做什么呢?

    Spring Boot的功能

    Spring Boot实现了自动配置,降低了项目搭建的复杂度。

    他本身并不提供Spring框架的核心特性以及扩展功能,只是用于快速、敏捷地开发新的基于Spring框架的应用程序。也就是说,它并不是用来替代Spring的解决方案,而是和Spring框架紧密结合用于提升Spring开发者体验的工具。同时呢,因为它集成了大量常用的第三方库配置(例如JDBC, Mongodb, Redis, Mail,rabbitmq等等),所以在Spring Boot应用中这些第三方库几乎可以零配置的开箱即用(out-of-the-box),大部分的Spring Boot应用都只需要非常少量的配置代码,开发者能够更加专注于业务逻辑。

    所以说,Spring Boot只是承载者,辅助你简化项目搭建过程的。如果承载的是WEB项目,使用Spring MVC作为MVC框架,那么工作流程和spring mvc是完全一样的,因为这部分工作是Spring MVC做的而不是Spring Boot。

    springboot与springmvc的关系
    二者区别

    所以说我们谈到springboot与spring mvc的关系区别的时候,有这么一句总结:

    Spring 是一个“引擎”;

    springmvc是框架,web项目中实际运行的代码;

    spring boot只是一个配置工具,整合工具,辅助工具。是一套快速开发整合包。

    Spring Boot :J2EE一站式解决方案

    Spring Cloud : 分布式整体解决方案

    Spring 最初利用“工厂模式”(DI)和“代理模式”(AOP)解耦应用组件。大家觉得挺好用,于是按照这种模式搞了一个 MVC框架(一些用Spring 解耦的组件),用开发 web 应用( SpringMVC )。然后发现每次开发都写很多样板代码,为了简化工作流程,于是开发出了一些“懒人整合包”(starter),这套就是 Spring Boot。

    springboot与微服务

    为什么说springboot与我们的微服务有关系呢?可以来看下springboot的一些特性~

    Spring Boot 特性
    • 使用 Spring 项目引导页面可以在几秒构建一个项目
    • 方便对外输出各种形式的服务,如 REST API、WebSocket、Web、Streaming、Tasks
    • 非常简洁的安全策略集成
    • 支持关系数据库和非关系数据库
    • 支持运行期内嵌容器,如 Tomcat、Jetty
    • 强大的开发包,支持热启动
    • 自动管理依赖
    • 自带应用监控
    • 支持各种 IED,如 IntelliJ IDEA 、NetBeans

    正是因为Spring Boot 的这些特性非常方便、快速构建独立的微服务。所以我们使用 Spring Boot 开发项目,会给我们传统开发带来非常大的便利度,可以说如果你使用过 Spring Boot 开发过项目,就不会再愿意以以前的方式去开发项目了。

     

    总结一下,使用 Spring Boot 至少可以给我们带来以下几方面的改进:

    • Spring Boot 使编码变简单,Spring Boot 提供了丰富的解决方案,快速集成各种解决方案提升开发效率。
    • Spring Boot 使配置变简单,Spring Boot 提供了丰富的 Starters,集成主流开源产品往往只需要简单的配置即可。
    • Spring Boot 使部署变简单,Spring Boot 本身内嵌启动容器,仅仅需要一个命令即可启动项目,结合 Jenkins 、Docker 自动化运维非常容易实现。
    • Spring Boot 使监控变简单,Spring Boot 自带监控组件,使用 Actuator 轻松监控服务各项状态。

    总结,Spring Boot 是 Java 领域最优秀的微服务架构落地技术,没有之一。

    springboot的三大特性

    组件自动装配:web mvc、jdbc、MongoDB

    嵌入式web容器:tomcat、jetty

    生产准备特性:指标、健康检查、外部化配置

    • 不影响功能开发,但是有指导意义
    • 应用是否监控的,微服务是否健康
    • 外部化配置:如server.port,可以改变服务端口等,以前在服务器上改端口
    切换web server

    tomcat切换成jetty,spring内置的tomcat容器

    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-web</artifactId>
        <exclusions><!-- Exclude the Tomcat dependency -->
            <exclusion>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-tomcat</artifactId>
            </exclusion>
        </exclusions>
    </dependency><!-- Use Jetty instead -->
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-jetty</artifactId>
    </dependency>
    认识springApplication
    main方法解释

    springboot项目的启动一般使用Java的main方法来启动项目,代码里面有个run方法,使用SpringBootDeepinApplication来作为参数,为什么要使用SpringBootDeepinApplication来作为参数,因为@SpringBootApplication也是@Configuration拓展注解,所以初始化的时候其实就是最先加载这个配置类,然后根据注解拓展开来加载各个模块。

    • 所以,参数不一定是主类。只要是有@SpringBootApplication注解的类都可以成为参数
    @SpringBootApplication
    public class SpringBootDeepinApplication {
    
    
        public static void main(String[] args) {
            SpringApplication.run(SpringBootDeepinApplication.class, args);
        }
    }
    boot的准备阶段

    类加载实现思路:

    1. 实现类
    1. 配置资源META-INF/spring.factories
    1. 排序:@AutoConfigureOrder,@Ordered,实现Ordered接口等方式
    • 加载spring.favorites的自动配置
    • 加载应用监听器

    首先SpringApplication使用主类作为配置参数初始化并加载资源。

    SpringApplication application = new SpringApplication(SpringBootDeepinApplication.class);

    每个模块中,只要classpath中有spring.favorites就可以装载有这个两个配置类的资源。

    #加载Spring.favorites中ApplicationContextInitializer的配置类
    this.setInitializers(this.getSpringFactoriesInstances(ApplicationContextInitializer.class));
    
    
    #加载Spring.favorites中ApplicationListener的配置类
    this.setListeners(this.getSpringFactoriesInstances(ApplicationListener.class));

    关键代码:

    #配置加载器,包括注解配置或xml配置
    BeanDefinitionLoader

    boot的运行阶段

    SpringApplication的run方法就是boot的运行阶段的,需要初始化环境,context,发布事件,启动监听器等等。

    spring的手动装配
    spring的模式注解装配

    @Component注解的拓展,“派生注解”-->@Controller、@Service、@Repository、分别和控制层(Web 层)、业务层和持久层相对应...

    模式注解的层次性

    spring的@Enable模块装配

    激活:@EnableAutoConfiguration

    配置:/META-INF/spring.factoryes

    实现:XXXAutoConfiguration

    框架实现

    @Enable 注解模块 激活模块

    Spring Framework

    @EnableWebMvc Web MVC 模块

     @EnableTransaction Management 事务管理模块

     @EnableCaching Caching 模块

     @EnableMBeanExport JMX 模块

     @EnableAsync 异步处理模块

     @EnableWebFlux Web Flux 模块

     @EnableAspectJAutoProxy AspectJ 代理模块

       

    Spring Boot

    @EnableAutoConfiguration 自动装配模块

     @EnableManagementContext Actuator 管理模块

     @EnableConfigurationProperties 配置属性绑定模块

     @EnableOAuth2Sso OAuth2 单点登录模块

       

    Spring Cloud

    @EnableEurekaServer Eureka服务器模块

     @EnableConfigServer 配置服务器模块

     @EnableFeignClients Feign客户端模块

     @EnableZuulProxy 服务网关 Zuul 模块

     @EnableCircuitBreaker 服务熔断模块

    自定义@Enable模块

    1、基于注解驱动实现,可参考@EnableWebMvc注解

    第一步:自定义@EnableXXX注解,其中@Import表示把用到的资源导入到当前容器中,相当于导入一个@Configuration配置类

    @Retention(RetentionPolicy.RUNTIME)
    @Target({ElementType.TYPE})
    @Documented
    @Import({SayHelloWorldConfiguration.class})
    public @interface EnableSayHelloWorld {
    
    
    }

    第二步、编写配置类,配置类中把需要的bean注入到spring容器中,因为已经有了第一步的@Import,所以这里不再需要@Configuration注解。

    //@Configuration
    public class SayHelloWorldConfiguration {
    
    
        @Bean
        SayHelloWorld sayHelloWorld() {
            System.out.println("here to loading bean sayhelloworld!");
            return new SayHelloWorld();
        }
    }
    //需要初始化的bean
    public class SayHelloWorld {
    
    
        public String say() {
            return "hello world";
        }
    
    
    }

    第三步、在启动项目时候加上@EnableSayHelloWorld注解,表示加载自定义的配置。@EnableSayHelloWorld是@Import的拓展注解,实际起作用的是@Import注解。

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

    2、基于接口驱动实现,参考@EnableCaching注解

    相比注解形式,接口驱动实现更加灵活,可以添加分支或根据条件判断来初始化程序。

    大概步骤:

    HelloWorldImportSelector -> HelloWorldConfiguration -> HelloWorld

    第一步、自定义注解@EnableSeletorHelloWorld

    @Target({ElementType.TYPE})
    @Retention(RetentionPolicy.RUNTIME)
    @Documented
    @Import({HelloWorldImportSeletor.class})
    public @interface EnableSeletorHelloWorld {
    
    
        String model() default "first";
    }

    第二步、编写Seletor,实现ImportSeletor这个借口,并重写SelectImports方法,返回的数组表示需要初始化注入到spring容器的中的bean名称。

    public class HelloWorldImportSeletor implements ImportSelector {
    
    
        @Override
        public String[] selectImports(AnnotationMetadata annotationMetadata) {
    
    
            //获取注解上的属性的值
            Map<String, Object> annotationAttributes = annotationMetadata.getAnnotationAttributes(EnableSeletorHelloWorld.class.getName());
            String model = (String) annotationAttributes.get("model");
            System.out.println(model);
    
    
            //可以返回多个加载的配置或bean
            return new String[]{SayHelloWorldConfiguration.class.getName()};
        }
    }

    其余与注解形式相同,这里略~

    spring条件装配

    定义:Bean装配的前置条件

    举例:@Profile、@Conditional

    1、注解方式

    @Profile 配置条件装配

    略~

    2、编程方式

    @Conditional 编程条件装配,

    可参考org.springframework.boot.autoconfigure.condition.ConditionalOnProperty

    第一步、编写条件注解。用于判断系统是否是linux,当linux的时候初始化对应的配置。

    @Retention(RetentionPolicy.RUNTIME)
    @Target({ElementType.TYPE, ElementType.METHOD})
    @Documented
    @Conditional({OnSystemPropertyCondition.class})
    public @interface ConditionalOnSystemProperty {
    
    
        String value();
    }

    第二步、编写条件判断类OnSystemPropertyCondition,实现org.springframework.context.annotation.Condition接口,重写match方法用于判断条件返回true或false;

    public class OnSystemPropertyCondition implements Condition {
    
    
        /**
         * 判断是否满足条件
         * @param conditionContext
         * @param annotatedTypeMetadata 注解的元信息
         * @return
         */
        @Override
        public boolean matches(ConditionContext conditionContext, AnnotatedTypeMetadata annotatedTypeMetadata) {
            Map<String, Object> attrs = annotatedTypeMetadata.getAnnotationAttributes(ConditionalOnSystemProperty.class.getName());
    
    
            String system = String.valueOf(attrs.get("value"));
            String currentOs = System.getProperty("os.name");
            return currentOs.endsWith(system);
        }
    }

    第三步、在需要做判断才能初始化的bean上添加此注解即可完成条件装配

    @ConditionalOnSystemProperty(value = "linux")
    springboot的自动装配模式
    Java SPI机制

    Java SPI是什么?

    SPI的全名为Service Provider Interface.大多数开发人员可能不熟悉,因为这个是针对厂商或者插件的。

    我们系统里抽象的各个模块,往往有很多不同的实现方案,比如日志模块的方案,xml解析模块、jdbc模块的方案等。

    面向的对象的设计里,我们一般推荐模块之间基于接口编程,模块之间不对实现类进行硬编码。一旦代码里涉及具体的实现类,就违反了可拔插的原则,如果需要替换一种实现,就需要修改代码。为了实现在模块装配的时候能不在程序里动态指明,这就需要一种服务发现机制。

    Java SPI就是提供这样的一个机制:为某个接口寻×××实现的机制。

    Java SPI的约定

    当服务的提供者,提供了服务接口的一种实现之后,在jar包的META-INF/services/目录里同时创建一个以服务接口命名的文件。

    该文件里就是实现该服务接口的具体实现类。而当外部程序装配这个模块的时候,就能通过该jar包META-INF/services/里的配置文件找到具体的实现类名,并装载实例化,完成模块的注入。

    基于这样一个约定就能很好的找到服务接口的实现类,而不需要再代码里制定。

    SpringBoot中的SPI机制

    在Spring中也有一种类似与Java SPI的加载机制。它在META-INF/spring.factories文件中配置接口的实现类名称,然后在程序中读取这些配置文件并实例化。

    这种自定义的SPI机制是Spring Boot Starter实现的基础。

    实现原则

    理念:约定大于配置原则

    装配:模式装配、@Enable注解装配、条件装配

    步骤:激活自动装配、实现自动装配、配置自动装配

    starter:启动器

    底层装配技术 
    • Spring 模式注解装配
    • Spring @Enable 模块装配
    • Spring 条件装配装配
    • Spring 工厂加载机制
    • 实现类: SpringFactoriesLoader (类似与Java SPI机制的一个加载类)
    • 配置资源: META-INF/spring.factories

    加载类SpringFactoriesLoader说明:

    spring的工厂加载类SpringFactoriesLoader,用于加载自动配置,主要方法loadFactories(),参数factoryClass表示要加载的工厂配置类,classLoader表示类加载器。

    配置文件META-INF/spring.factories说明:

    自定义自动条件装配

    步骤:

    • HelloWorldAutoConfiguration 
    • 条件判断: user.name == "Mercy"
    • 模式注解: @Configuration @Enable 模块: @EnableHelloWorld -> HelloWorldImportSelector -> HelloWorldConfiguration - > helloWorld

    查看这个类WebMvcAutoConfiguration,可以看到这个配置类就用了条件装配。

    @Configuration
    @ConditionalOnWebApplication(
        type = Type.SERVLET
    )
    @ConditionalOnClass({Servlet.class, DispatcherServlet.class, WebMvcConfigurer.class})
    @ConditionalOnMissingBean({WebMvcConfigurationSupport.class})
    @AutoConfigureOrder(-2147483638)
    @AutoConfigureAfter({DispatcherServletAutoConfiguration.class, ValidationAutoConfiguration.class})
    public class WebMvcAutoConfiguration {
     ...
    }
    
    

    @ConditionalOnMissingBean表示当容器中没有这个bean的时候

    @ConditionalOnClass表示对应的类在classpath目录下存在时

    @ConditionalOnWebApplication表示当前项目是个servlet项目的时候

    第一步、编写需要自动装载的配置类,因为@Configuration注解的存在,因此我把需要自动装载的配置类放在了springboot扫描不到的包上。

    说明:

    @Configuration表示是个配置类

    @ConditionalOnSystemProperty表示需要满足当前系统是win10系统

    @EnableSeletorHelloWorld表示启动这个模块的加载

    @Configuration
    @ConditionalOnSystemProperty(value = "Windows 10")
    @EnableSeletorHelloWorld
    public class SayHelloWorldAutoConfiguration {
    
    
    
    
        @Bean
        SayHelloWorld autoSayHelloWorld() {
            System.out.println("here to !!auto!! loading bean autoSayHelloWorld!");
            return new SayHelloWorld();
        }
    }

    第二步、在resources目录下新建META-INF文件夹,编写spring.factories。

    # Auto Configure 自动装配自定义的配置
    org.springframework.boot.autoconfigure.EnableAutoConfiguration=
    com.other.configuration.SayHelloWorldAutoConfiguration

    启动springboot之后就会自动加载这个配置类~

    第二天的课程内容:

    springboot的异常处理
    springboot集成redis
    springboot集成spring session
    springboot的打包方式
     
  • 相关阅读:
    ElasticSearch实战系列一: ElasticSearch集群+Kinaba安装教程
    SpringBoot事物Transaction实战讲解教程
    SpringBoot整合Swagger和Actuator
    SpringBoot项目实现文件上传和邮件发送
    SpringBoot优雅的全局异常处理
    SpringCloud学习系列之七 ----- Zuul路由网关的过滤器和异常处理
    SpringBoot整合Redis使用Restful风格实现CRUD功能
    SpringCloud学习系列之六 ----- 路由网关Zuul基础使用教程
    SpringCloud学习系列之五-----配置中心(Config)和消息总线(Bus)完美使用版
    SpringCloud学习系列之四-----配置中心(Config)使用详解
  • 原文地址:https://www.cnblogs.com/free-wings/p/9795796.html
Copyright © 2020-2023  润新知