Spring boot 2.0 消息总线(Spring Cloud Bus) 高可用分布式配置中心 实例

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Spring boot 2.0 消息总线(Spring Cloud Bus) 高可用分布式配置中心 实例相关的知识,希望对你有一定的参考价值。

参考技术A 配置中心的客户端要使用bootstrap.yml
bootstrap.yml优先于 application.yml 加载 即 bootstrap.yml加载完后才加载 application.yml

在需要更新的配置类上加@RefreshScope注解,@RefreshScope必须加,否则客户端会收到服务端的更新消息,但是更新不了,因为不知道更新哪里的。

18082: 是统一配置中心的端口
/auth-server/dev 对应远程配置文件的auth-server-dev.yml
/name-profiles.yml
/label/name-profiles.yml
name : 文件名,一般以服务名来命名
profiles : 一般作为环境标识
lable : 分支(branch),指定访问某分支下的配置文件

业务微服端的测试接口: http://127.0.0.1:18088/test

修改配置文件并提交到git上

我们发现再不重启服务的情况下就能获取到修改的属性值

只要开启 Spring Cloud Bus 后,不管是对 config-server 还是 config-client 执行/actuator/bus-refresh都是可以更新配置的,
如果有多个客户端,多个客户端都会接收到刷新配置的消息,并刷新配置。

某些场景下(例如灰度发布),我们可能只想刷新部分微服务的配置,此时可通过/actuator/bus-refresh/destination端点的 destination 参数来定位要刷新的应用程序。例如:/actuator/bus-refresh/customers:8000,这样消息总线上的微服务实例就会根据 destination 参数的值来判断是否需要要刷新。其中,customers:8000指的是各个微服务的 ApplicationContext ID。destination 参数也可以用来定位特定的微服务。例如:/actuator/bus-refresh/customers:**,这样就可以触发 customers 微服务所有实例的配置刷新。

远程配置文件修改后,不需要手动调用 http://localhost:18088/actuator/bus-refresh 即可刷新配置

选择Just the push event.,因为我们只需要push的时候进行回调,然后添加即可

每当我们更新远程配置文件后,push到git后,就会触发Payload URL(配置的是访问刷新配置的地址)

(转)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的打包方式
 

以上是关于Spring boot 2.0 消息总线(Spring Cloud Bus) 高可用分布式配置中心 实例的主要内容,如果未能解决你的问题,请参考以下文章

springboot 业务消息总线及延时消息回调

阿里云的消息服务怎么集成spring boot

Spring Boot 揭秘与实战 服务器篇 - 其他内嵌服务器 发表于 2017-01-03 | Spring框架 | Spri

Spring Cloud Azure 参考文档

Spring Cloud

SpringCloud-2.0-周阳(14. 消息总线 - SpringCloud Bus)