-
1.项目Gitee地址https://gitee.com/caichunfu/dtse-practice-microservice2.运行环境JDK1.8Maven3.6.3本地CSE引擎下载地址:https://support.huaweicloud.com/devg-cse/cse_devg_0036.html3.注意事项踩过的一些坑:3.1依赖导入报错需要把Maven的中央仓库地址改为华为中央仓库地址,修改setting.xml文件 <mirror> <id>huaweicloud</id> <mirrorOf>*,!HuaweiCloudSDK</mirrorOf> <url>https://repo.huaweicloud.com/repository/maven/</url> </mirror>3.2版本问题官网地址:https://github.com/huaweicloud/spring-cloud-huawei1)使用Hoxton分支,项目启动时ribbon的启动类会报错,建议使用master分支com.huaweicloud.servicecomb.discovery.ribbon.ServiceCombRibbonClientConfiguration required a bean of type 'com.netflix.client.config.IClientConfig' that could not be found.2)master分支1.8.0-2020.0.x与网关存在兼容性问题,建议使用1.6.1-2020.0.x版本3)master分支需要使用 springcloud 2020.x 的版本,这个版本移除了Netflix相关的组件,所以需要对组件进行替换3.3本次测试使用的版本 <properties> <spring-boot.version>2.5.3</spring-boot.version> <spring-cloud.version>2020.0.4</spring-cloud.version> <spring-cloud-huawei.version>1.6.1-2020.0.x</spring-cloud-huawei.version> <servicecomb.version>2.5.0</servicecomb.version> <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> </properties>4.引入依赖4.1父工程引入 <dependencyManagement> <!-- configure spring cloud huawei version --> <dependency> <groupId>com.huaweicloud</groupId> <artifactId>spring-cloud-huawei-bom</artifactId> <version>${spring-cloud-huawei.version}</version> <type>pom</type> <scope>import</scope> </dependency> </dependencies> </dependencyManagement>4.2子工程引入,并删除eureka相关依赖<dependency> <groupId>com.huaweicloud</groupId> <artifactId>spring-cloud-starter-huawei-service-engine</artifactId></dependency>5.创建bootstrap.yml文件spring: application: name: #微服务名 cloud: servicecomb: discovery: enabled: true address: http://127.0.0.1:30100 appName: #应用名 serviceName: ${spring.application.name} version: 0.0.1 healthCheckInterval: 30 config: serverAddr: http://127.0.0.1:30110 serverType: kie6.启动类添加注解@EnableDiscoveryClient启动本地微服引擎,访问http://127.0.0.1:30103/进入微服务引擎管理界面,查看服务是否注册成功7.启动前遇到的问题7.1报错信息:could not be registered. A bean with that name has already been defined in URL Action:Consider renaming one of the beans or enabling overriding by setting spring.main.allow-bean-definition-overriding=true无法注册。URL中已经定义了一个具有该名称的bean考虑重命名一个bean,或者通过设置Spring实现重写。主要的允许bean定义覆盖=true解决方案:yml配置文件添加配置spring: main: allow-bean-definition-overriding: true7.2报错信息:Action:Correct the classpath of your application so that it contains a single, compatible version of io.micrometer.core.instrument.distribution.DistributionStatisticConfig$Builder解决方案:pom.xml里面的依赖包有重复,需要将重复的依赖包删除7.3报错信息:此问题只会在使用Honxton版本时出现,建议使用master分支1.6.1-2020.0.x版本Parameter 0 of method ribbonServerList in com.huaweicloud.servicecomb.discovery.ribbon.ServiceCombRibbonClientConfiguration required a bean of type 'com.netflix.client.config.IClientConfig' that could not be found.Action:Consider defining a bean of type 'com.netflix.client.config.IClientConfig' in your configuration.解决方案:IClientConfig 类,这个类定义时,不能直接 return 一个没有任何属性的 DefaultClientConfigImpl 对象, openFeign 会在源码里面使用这个对象,报空指针异常,如果要自己定义,需要初始化里面该有的属性@Configurationpublic class IClientConfig { @Bean public DefaultClientConfigImpl iClientConfig(){ //网上很多的错误写法 //return new DefaultClientConfigImp(); //加上getClientConfigWithDefaultValues初始化参数 return DefaultClientConfigImpl.getClientConfigWithDefaultValues(); }}8.Feign远程调用使用SpringCloudHuawei做远程调用时会报错,可能兼容性存在问题为了验证问题还原了SpringCloud项目,openfeign调用不会报错报错信息:org.apache.servicecomb.service.center.client.exception.OperationException: get service instances list fails, statusCode = 400; message = Bad Request; content = {"errorCode":"400012","errorMessage":"Micro-service does not exist","detail":"Consumer[30b75156753bc55385a7ae74d0611c77fc5f7522][development/dtse-practice-microservice/dtse-system/0.0.1] find provider[development/dtse-practice-microservice/default/0+] failed, provider does not exist"}Servlet.service() for servlet [dispatcherServlet] in context with path [] threw exception [Request processing failed; nested exception is java.lang.RuntimeException: com.netflix.client.ClientException: Load balancer does not have available server for client: default解决---暂未解决:已对接后端技术专家,暂未解决,解决后会更新进度9.改为RestTemplate方式调用前端参数为MultipartFile和JSON,请求类型为POST @PostMapping(value = "/upload", consumes = MediaType.MULTIPART_FORM_DATA_VALUE) public String upLoadOneFile(@RequestPart("file") MultipartFile file, @RequestParam("obsParamsJson") String obsParamsJson) throws IOException { OBSStorageParams obsParams = JSON.parseObject(obsParamsJson, OBSStorageParams.class); String objURL = obsService.uploadOneFile(file.getInputStream(), obsParams); return objURL; }RestTemplate调用代码 //请求url String url = "http://xxxx:xxxx/xx/x"; //构造请求头 HttpHeaders httpHeaders = new HttpHeaders(); HttpHeaders headers = httpHeaders; headers.setContentType(MediaType.MULTIPART_FORM_DATA); //FileSystemResource将文件变成流以发送 File file = MultipartFileToFile.multipartFileToFile(multipartFile); FileSystemResource fileSystemResource = new FileSystemResource(file); //构造请求体,使用LinkedMultiValueMap MultiValueMap<String, Object> resultMap = new LinkedMultiValueMap<>(); resultMap.add("file", fileSystemResource); resultMap.add("obsParamsJson", obsParamsJson); //HttpEntity封装整个请求报文 HttpEntity<MultiValueMap<String, Object>> httpEntity = new HttpEntity<>(resultMap, headers); //postForObject发送请求体 String objURL = restTemplate.postForObject(url, httpEntity, String.class);MultipartFile转File import org.springframework.web.multipart.MultipartFile; import java.io.File;import java.io.FileOutputStream;import java.io.InputStream;import java.io.OutputStream;public class MultipartFileToFile { /** * MultipartFile 转 File * * @param file * @throws Exception */ public static File multipartFileToFile(MultipartFile file) throws Exception { File toFile = null; if (file.equals("") || file.getSize() <= 0) { file = null; } else { InputStream ins = null; ins = file.getInputStream(); toFile = new File(file.getOriginalFilename()); inputStreamToFile(ins, toFile); ins.close(); } return toFile; } //获取流文件 private static void inputStreamToFile(InputStream ins, File file) { try { OutputStream os = new FileOutputStream(file); int bytesRead = 0; byte[] buffer = new byte[8192]; while ((bytesRead = ins.read(buffer, 0, 8192)) != -1) { os.write(buffer, 0, bytesRead); } os.close(); ins.close(); } catch (Exception e) { e.printStackTrace(); } } /** * 删除本地临时文件 * @param file */ public static void delteTempFile(File file) { if (file != null) { File del = new File(file.toURI()); del.delete(); }}}10.网关改造官网的说明:Spring Cloud Huawei Hoxton分支只提供Spring Cloud Gateway基于Ribbon的负载均衡,及其配套的基于流量治理和灰度发布功能。 Spring Cloud Huawei master(2020.0.x版本)分支只提供Spring Cloud Gateway基于Spring Cloud LoadBalance的负载均衡, 及其配套的基于流量治理和灰度发布功能。建议Spring Cloud Gateway升级到2020.0.x版本。由于原项目使用的网关为Zuul,需要改为Spring Cloud Gateway10.1删除原项目zuul的依赖,添加springcloudgateway的依赖和springcloudhuawei提供的网关依赖<dependency> <groupId>org.springframework.cloud</groupId> <artifactId>spring-cloud-starter-gateway</artifactId></dependency><dependency> <groupId>com.huaweicloud</groupId> <artifactId>spring-cloud-starter-huawei-service-engine-gateway</artifactId></dependency>10.2添加网关配置文件spring: main: allow-bean-definition-overriding: true cloud: gateway: # 路由(routes:路由,它由唯一标识(ID)、目标服务地址(uri)、一组断言(predicates)和一组过滤器组成(filters)。filters 不是必需参数。) routes: #路由标识(id:标识,具有唯一性) - id: dtse-system-route # 目标服务地址(uri:地址,请求转发后的地址) uri: lb://dtse-system filters: args: # 路由条件(predicates:断言,匹配 HTTP 请求内容) predicates: - Path=/**urifiler: login-uri: /login10.3定义全局过滤器实现鉴权package com.huaweicloud.filter;import com.huaweicloud.commons.outhUtils.JwtUtil;import com.huaweicloud.commons.response.ResultCode;import com.huaweicloud.config.URIFilter;import io.jsonwebtoken.Claims;import lombok.extern.slf4j.Slf4j;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.cloud.gateway.filter.GatewayFilterChain;import org.springframework.cloud.gateway.filter.GlobalFilter;import org.springframework.core.Ordered;import org.springframework.http.server.RequestPath;import org.springframework.http.server.reactive.ServerHttpRequest;import org.springframework.stereotype.Component;import org.springframework.util.StringUtils;import org.springframework.web.server.ServerWebExchange;import reactor.core.publisher.Mono;@Component@Slf4jpublic class RouteConfiguration implements GlobalFilter, Ordered { @Autowired JwtUtil jwtUtil; @Autowired URIFilter uriFilter; @Override public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) { ServerHttpRequest request = exchange.getRequest(); RequestPath path = request.getPath(); // 2、登陆请求放行 if(path.value().contains(uriFilter.getLoginuri().get(0))){ System.out.println("登陆请求路经:request.getPath() = " + path.value()); log.info("登录"); return chain.filter(exchange); } //3、非登陆请求用户权限校验 String authorization = request.getHeaders().getFirst("Authorization"); if (!StringUtils.isEmpty((authorization))) { System.out.println("非登陆请求路径:request.getPath() = " + path.value()); //2、获取请求头中Token信息 String token = authorization.replace("Bearer", ""); //3、Token校验 Claims claims = jwtUtil.parseToken(token) ; //4、获取用户id,并将用户id传送到后端 if (claims == null) { try { throw new Exception(String.valueOf(ResultCode.UNAUTHENTICATED)); } catch (Exception e) { e.printStackTrace(); } return null; } String id = claims.getId(); //5、添加用户请求头 request.mutate().header("userId",id).build(); return chain.filter(exchange); } return chain.filter(exchange); } @Override public int getOrder() { return 0; }}10.4urifiler配置类@ConfigurationProperties(prefix = "urifiler", ignoreUnknownFields = false)@Data@Componentpublic class URIFilter { private List<String> loginuri;}11.配置中心的使用访问http://127.0.0.1:30103/进入微服务本地引擎管理界面选择配置列表,创建配置项
-
1.面临问题:session存在于服务端,分布式中多个微服务,每个微服务都是独立的服务器,session信息不同步,该怎么办?2.解决思路 token替代session使用流程:1.用户登录成功 生成token存入redis(key:token,value:用户信息);把token返回给前端2.将token加入header,使用过滤器,在每个接口执行前验证token是否有效,无效返回登录页3.微服务中如果想获取登录的用户信息,可以获取redis的key=tokentoken + redis优点:1.分布式session,解决同步问题2.单点登录 和(确保系统同一账号登录)3.token统一管理,响应快
-
目 录... 21 使用前必读... 31.1 概述... 31.2 基本概念... 32 迁移步骤... 42.1 目标... 42.2 必要配置修改... 42.3 SpringCloud替换点... 52.4 修改前后对比... 63 验证... 73.1 微服务Istio集群部署... 73.2 微服务间访问... 73.3 主要治理能力... 83.3.1 灰度发布... 83.3.2 应用拓扑... 93.3.3 会话保持负载均衡... 9 1 使用前必读1.1 概述本教程介绍传统使用Spring Cloud 开发的微服务,做容器化改造,直接使用 Kubernetes +Istio 的方案使用基础设施提供的运行治理能力。1.2 基本概念容器与Docker容器技术起源于Linux,是一种内核虚拟化技术,提供轻量级的虚拟化,以便隔离进程和资源。尽管容器技术已经出现很久,却是随着Docker的出现而变得广为人知。Docker是第一个使容器能在不同机器之间移植的系统。它不仅简化了打包应用的流程,也简化了打包应用的库和依赖,甚至整个操作系统的文件系统能被打包成一个简单的可移植的包,这个包可以被用来在任何其他运行Docker的机器上使用。KubernetesKubernetes是一个很容易地部署和管理容器化的应用软件系统,使用Kubernetes能够方便对容器进行调度和编排。对应用开发者而言,可以把Kubernetes看成一个集群操作系统。Kubernetes提供服务发现、伸缩、负载均衡、自愈甚至选举等功能,让开发者从基础设施相关配置等解脱出来。Kubernetes可以把大量的服务器看做一台巨大的服务器,在一台大服务器上面运行应用程序。无论Kubernetes的集群有多少台服务器,在Kubernetes上部署应用程序的方法永远一样。PodPod是Kubernetes创建或部署的最小单位。一个Pod封装一个或多个容器(container)、存储资源(volume)、一个独立的网络IP以及管理控制容器运行方式的策略选项。ASM以基础设施的方式为用户提供服务流量管理、服务运行监控、服务访问安全以及服务发布能力。控制面和数据面均和开源Istio完全兼容,无缝对接了华为云的企业级Kubernetes集群服务云容器引擎CCE,可为客户提供开箱即用的上手体验IstioIstio是一个提供连接、保护、控制以及观测功能的开放平台,通过提供完整的非侵入式的微服务治理解决方案,能够很好的解决云原生服务的管理、网络连接以及安全管理等服务网络治理问题。2 迁移步骤2.1 目标尽可能少的修改,将原来SpringCloud开发的微服务迁移到Kubernetes Istio上来。2.2 必要配置修改对SpringCloud 微服务做适当修改,可以将Spring Cloud的微服务的流量导流到Isito的代理商,从而执行Istio的服务治理能力。代码无需修改,只需要修改微服务的配置文件application.yml即可。① 掉对Eureka连接配置,不再连接Eureka注册中心。② 配置调用的微服务信息③ 禁用Eureka服务注册,服务发现,禁用Hystrix等。其中①和③都是通用的禁用SpringCloud中原有微服务治理的功能,只有②是必要的对要访问的服务进行适当配置。2.3 SpringCloud替换点下表根据SpringCloud官方https://cloud.spring.io/spring-cloud-static/spring-cloud-netflix/2.2.0.提供的服务治理能力在迁移到Istio后的对应处理。服务发现、负载均衡等完全废弃熔断等能力可选禁用掉使用Istio的对应能力。2.4 修改前后对比如下比较两个微服务项目的全部文件,可以看到迁移到Istio上只需修改两个微服务的配置文件即可。3 验证省略代码制作镜像和在 k8s 集群上部署 微服务 的前置操作,详细参照 CCE 微服务上云指导。3.1 微服务Istio集群部署只有两个微服务helloclient和helloserver没有eureka服务集群启用了Istio;负载都注入了Sidecar;3.2 微服务间访问目标服务有三个实例,pod名和PodIp分别如图:通过k8s 服务名访问 client 微服务curl http://helloclient.springcloud-passthrough.svc.cluster.local:7211/hello观察client微服务侧的Envoy日志kubectl logs helloclient-6fcc9cb8c9 qz5ng -c istio-proxy –nspringcloud-passthrough -f可以看到从client到server的每个请求的outbound流量都经过了client端的Envoy从而接管了SpringCloud微服务的服务发现和负载均衡。因此所有Isito的治理规则都可以在该微服务上定义和执行。3.3 主要治理能力3.3.1 灰度发布3.3.2 应用拓扑3.3.3 会话保持负载均衡
-
2020-12-09 15:04:27,038 [WARN] method GET, path /baseInfoExport/gl_exportBaseInfoExcel/, statusCode 200, reasonPhrase , response content-type application/vnd.ms-excel is not supported org.apache.servicecomb.transport.rest.client.http.DefaultHttpClientFilter.extractResponse(DefaultHttpClientFilter.java:99)2020-12-09 15:04:27,039 [ERROR] failed to decode response body, exception is [Invalid UTF-8 middle byte 0x11 at [Source: (org.apache.servicecomb.foundation.vertx.stream.BufferInputStream); line: 1, column: 4]] org.apache.servicecomb.transport.rest.client.http.DefaultHttpClientFilter.extractResponse(DefaultHttpClientFilter.java:107)2020-12-09 15:04:34,783 [INFO] create MicroserviceVersions, appId=ctjsoft, microserviceName=portalet-server2. org.apache.servicecomb.serviceregistry.consumer.MicroserviceVersions.<init>(MicroserviceVersions.java:83)
-
①. 服务发现——Netflix Eureka一个RESTful服务,用来定位运行在AWS地区(Region)中的中间层服务。由两个组件组成:Eureka服务器和Eureka客户端。Eureka服务器用作服务注册服务器。Eureka客户端是一个java客户端,用来简化与服务器的交互、作为轮询负载均衡器,并提供服务的故障切换支持。Netflix在其生产环境中使用的是另外的客户端,它提供基于流量、资源利用率以及出错状态的加权负载均衡。②. 客服端负载均衡——Netflix RibbonRibbon,主要提供客户侧的软件负载均衡算法。Ribbon客户端组件提供一系列完善的配置选项,比如连接超时、重试、重试算法等。Ribbon内置可插拔、可定制的负载均衡组件。③. 断路器——Netflix Hystrix断路器可以防止一个应用程序多次试图执行一个操作,即很可能失败,允许它继续而不等待故障恢复或者浪费 CPU 周期,而它确定该故障是持久的。断路器模式也使应用程序能够检测故障是否已经解决。如果问题似乎已经得到纠正,应用程序可以尝试调用操作。④. 服务网关——Netflix Zuul类似nginx,反向代理的功能,不过netflix自己增加了一些配合其他组件的特性。⑤. 分布式配置——Spring Cloud Config这个还是静态的,得配合Spring Cloud Bus实现动态的配置更新。来自:https://blog.csdn.net/fangchao2011/article/details/89186765
-
云社区 博客 博客详情spring cloud 2.0 概述 【摘要】 传统单体架构就是单点应用,也就是在早期开发学习的ssm或ssh整合项目采用分层架构模式、数据库访问层、业务逻辑层、控制层,从前端到后端所有代码都是一个人写的微服务架构演变过程传统单体架构 =》 分布式架构 =》 soa面向服务架构 =》 微服务架构传统单体架构传统单体架构就是单点应用,也就是在早期开发学习的ssm或ssh整合项目采用分层架构模式、数据库访问层、业务逻辑层、控制层,从前端到后端所有代码都是一个人写的cn.itycu.controler ---springmvc 视图层 jsp/ftl cn.itycu.service ---业务逻辑层 cn.itycu.dao ---数据库访问层 将所有的代码都放入到同一个项目中,部署在同一个tomcat中该架构模式存在的优缺点优点:开发简单、运维简单缺点:该架构模式没有对业务逻辑进行拆分,这样子耦合度非常高,只适合小团队或者个人形式开发,不适合团队模式协同工作开发,维护性很难,如果系统中某个模块出现问题,会导致整个系统无法使用。应用场景:政府项目、管理系统、crm、oa,适合于小团队或个人进行开发分布式架构分布式架构模式基于传统的架构模式演变过来,将我们传统的单点系统实现根据业务拆分。会拆分为会员系统、订单系统、支付系统、秒杀系统等。从而降低整个项目的耦合度,这种架构模式开始慢慢适合于互联网公司开发。会员系统 memner.itycu.cn 支付系统 pay.itycu.cn 项目命名为系统意味:包含视图层和服务层soa面向服务架构sso单点登录系统,抽离出通用服务soa面向服务架构基于分布式架构模式演变过来,俗称服务化,也就是面向于服务与接口开发(服务开发),将共同存在的业务逻辑抽取成一个公共服务,提供给其他接口实现调用,服务与服务之间采用rpc远程调用技术。服务:只是有接口、没有视图层cn.itycu.service cn.itycu.dao能够解决我们的代码冗余性问题soa架构模式特点传统政府、银行项目还是保留的在使用webservicewebservice架构模式wsdl组件表示接口信息、方法、调用地址、参数soa架构模式传输协议采用soap协议(HTTP/https+xml)实现传输,在高并发情况下实现通讯协议存在大量的冗余性传输,而且非常占用带宽。所以后来微服务架构中使用json替代了xml。soa架构模式实现方案webservice或者esb企业服务总线,底层采用soap传输协议。soa架构模式存在缺点前后端分离就是对我们控制层和业务逻辑实现区分,前端控制可以采用vue调用我们后端接口(http+json)采用soap协议实现通讯,xml传输非常重,效率比较低。服务化管理和治理设施不够完善依赖于中心服务发现机制不适合前后端分离架构模式微服务架构微服务架构模式基于soa架构模式演变过来的,比soa架构迷失对服务拆分力度会更加精细,采用前后端分离模式让专业的人做专业的事(专注),目的可以实现高效率开发。微服务架构中,每个服务之间都是互不影响,每个服务必须要独立部署、运维、互不影响,微服务架构模式非常轻巧,轻量级、适合于互联网公司开发模式。服务与服务之间通讯的协议采用restful形式,数据交换格式采用http+json格式实现传输整个过程中,采用二进制,所以http协议可以实现跨语言的平台,并且和其他语言实现通讯,所以为什么开放都是采用http+json格式传输SOA架构与微服务架构有哪些区别微服务架构模式比soa架构模式,更加适合于互联网公司敏捷、高效、快速迭代版本开发,因为力度非常精细。微服务架构模式比soa架构模式拆分力度更加精细,提倡让专业的人做专业的事,目的是实现高效的开发,每个服务与服务之间互不影响,每个服务都是单独独立数据库、redis连接、MQ等。并且都是实现独立部署,整个微服务架构更加轻量级。在soa架构中,有可能纯在多个服务共享同一个数据库,但是微服务架构必须强调每个都是独立数据库部署,互不影响。微服务架构基于soa架构模式演变过来,继承了soa架构的优点,在微服务架构中去除soa架构中soap协议和esp企业服务总线。改为http+json形式传输我们的数据。esb企业服务总线解决多系统间跨语言无法实现通讯的问题,对我们的数据实现转换,可以提供可靠的消息传输。一般情况我们采用http+json传输数据,不需要esb对数据进行转换。通讯协议服务拆分力度迭代微服务架构中可能会存在哪些问题分布式事务解决方案(rabbitmq、rocketmq事务消息、lcn(淘汰)、setata)最终一级概念分布式任务调度平台(xxl-job、elastic-job)分布式服务注册与发现(eureka、consul、zookeeper、nacos)分布式日志采集系统elk+kafka分布式服务最综与调用链系统zipkin分布式服务配置中心(spring cloud config/携程阿波罗/nacos/disconfig)微服务架构中有个非常重要的概念:独立部署、可配置、动态化。为什么要使用到spring cloudspring cloud 并不是一个rpc远程调用框架,而是一个微服务全家桶的解决方案的框架。理念就是***服务解决我们在微服务架构中遇到的问题。服务治理:eureka分布式配置:config客户端调用工具:rest/feign客户 rpc远程调用说明:阿里巴巴、腾讯、百度注意:大家如果去一些比较大型的互联网公司中,整个公司内部实现rpc通讯的框架、服务帮助治理都是内部自己研发。rpc远程调用框架:HTTPclent、dubbo、feign、grpc、基于netty手写rpcspring cloud一代和二代的区别spring cloud第一代实际上采用Netflix开源的组件整合微服务解决方案。spring cloud第二代实际上就是自己研发和国内的优秀的服务解决微服务框架进行组合。nacos实现服务注册于发现微服务服务治理核心概念Nacos产生背景rpc远程调用框架:HTTP client、grpc、dubbo、rest、openfeign等。传统rpc远程调用中存在哪些问题?超时、安全、服务与服务之间的url地址管理在我们的微服务架构通讯中,服务间依赖非常大,如果使用传统方式管理我们的服务,非常麻烦,所以采用url治理技术,可以实现我们整个实现动态服务注册与发现、本地负载均衡、容错等nacos分布式注册与发现功能 | 分布式配置中心产生于rpc远程调用中,服务的url的治理传统服务注册中心的实现方式把每个服务器的地址信息和端口号人工存放到数据库表中IdserviceIP端口号1itycu.cn192.168.1.180802itycu.cn192.168.1.18000维护成本高没有实现动态自动化基于数据库形式实现服务url治理的缺点分布式注册中心实现原理注册中心的作用管理整个微服务url地址可以实现动态感知注册中心:duboo依赖zookeeper、eureka、consul、nacos、redis、数据库nacos与eureka的区别eureka与zookeeper的区别注册中心的原理keyIP端口号服务名称192.168.1.18080生产者启动时,根据这种存储方式注册到微服务注册中心根据以上存储方式的服务名称获取到IP地址和端口号获取到地址后在本地实现rpc远程调用生产者:提供我们接口被其他服务调用消费者:调用接口实现服务服务注册:提供服务接口地址信息存放Naxos基本介绍实现注册中心和分布式配置中心默认账号密码:nacos使用命令模式实现对nacos的注册使用discovertyClient从注册中心获取接口地址使用rest Template实现rpc远程调用纯手写本地负载均衡轮询算法实现线下服务动态感知
-
微服务架构演变过程传统单体架构 =》 分布式架构 =》 soa面向服务架构 =》 微服务架构传统单体架构传统单体架构就是单点应用,也就是在早期开发学习的ssm或ssh整合项目采用分层架构模式、数据库访问层、业务逻辑层、控制层,从前端到后端所有代码都是一个人写的cn.itycu.controler ---springmvc 视图层 jsp/ftl cn.itycu.service ---业务逻辑层 cn.itycu.dao ---数据库访问层 将所有的代码都放入到同一个项目中,部署在同一个tomcat中该架构模式存在的优缺点优点:开发简单、运维简单缺点:该架构模式没有对业务逻辑进行拆分,这样子耦合度非常高,只适合小团队或者个人形式开发,不适合团队模式协同工作开发,维护性很难,如果系统中某个模块出现问题,会导致整个系统无法使用。应用场景:政府项目、管理系统、crm、oa,适合于小团队或个人进行开发分布式架构分布式架构模式基于传统的架构模式演变过来,将我们传统的单点系统实现根据业务拆分。会拆分为会员系统、订单系统、支付系统、秒杀系统等。从而降低整个项目的耦合度,这种架构模式开始慢慢适合于互联网公司开发。会员系统 memner.itycu.cn 支付系统 pay.itycu.cn 项目命名为系统意味:包含视图层和服务层soa面向服务架构sso单点登录系统,抽离出通用服务soa面向服务架构基于分布式架构模式演变过来,俗称服务化,也就是面向于服务与接口开发(服务开发),将共同存在的业务逻辑抽取成一个公共服务,提供给其他接口实现调用,服务与服务之间采用rpc远程调用技术。服务:只是有接口、没有视图层cn.itycu.service cn.itycu.dao能够解决我们的代码冗余性问题soa架构模式特点传统政府、银行项目还是保留的在使用webservicewebservice架构模式wsdl组件表示接口信息、方法、调用地址、参数soa架构模式传输协议采用soap协议(HTTP/https+xml)实现传输,在高并发情况下实现通讯协议存在大量的冗余性传输,而且非常占用带宽。所以后来微服务架构中使用json替代了xml。soa架构模式实现方案webservice或者esb企业服务总线,底层采用soap传输协议。soa架构模式存在缺点前后端分离就是对我们控制层和业务逻辑实现区分,前端控制可以采用vue调用我们后端接口(http+json)采用soap协议实现通讯,xml传输非常重,效率比较低。服务化管理和治理设施不够完善依赖于中心服务发现机制不适合前后端分离架构模式微服务架构微服务架构模式基于soa架构模式演变过来的,比soa架构迷失对服务拆分力度会更加精细,采用前后端分离模式让专业的人做专业的事(专注),目的可以实现高效率开发。微服务架构中,每个服务之间都是互不影响,每个服务必须要独立部署、运维、互不影响,微服务架构模式非常轻巧,轻量级、适合于互联网公司开发模式。服务与服务之间通讯的协议采用restful形式,数据交换格式采用http+json格式实现传输整个过程中,采用二进制,所以http协议可以实现跨语言的平台,并且和其他语言实现通讯,所以为什么开放都是采用http+json格式传输SOA架构与微服务架构有哪些区别微服务架构模式比soa架构模式,更加适合于互联网公司敏捷、高效、快速迭代版本开发,因为力度非常精细。微服务架构模式比soa架构模式拆分力度更加精细,提倡让专业的人做专业的事,目的是实现高效的开发,每个服务与服务之间互不影响,每个服务都是单独独立数据库、redis连接、MQ等。并且都是实现独立部署,整个微服务架构更加轻量级。在soa架构中,有可能纯在多个服务共享同一个数据库,但是微服务架构必须强调每个都是独立数据库部署,互不影响。微服务架构基于soa架构模式演变过来,继承了soa架构的优点,在微服务架构中去除soa架构中soap协议和esp企业服务总线。改为http+json形式传输我们的数据。esb企业服务总线解决多系统间跨语言无法实现通讯的问题,对我们的数据实现转换,可以提供可靠的消息传输。一般情况我们采用http+json传输数据,不需要esb对数据进行转换。通讯协议服务拆分力度迭代微服务架构中可能会存在哪些问题分布式事务解决方案(rabbitmq、rocketmq事务消息、lcn(淘汰)、setata)最终一级概念分布式任务调度平台(xxl-job、elastic-job)分布式服务注册与发现(eureka、consul、zookeeper、nacos)分布式日志采集系统elk+kafka分布式服务最综与调用链系统zipkin分布式服务配置中心(spring cloud config/携程阿波罗/nacos/disconfig)微服务架构中有个非常重要的概念:独立部署、可配置、动态化。为什么要使用到spring cloudspring cloud 并不是一个rpc远程调用框架,而是一个微服务全家桶的解决方案的框架。理念就是***服务解决我们在微服务架构中遇到的问题。服务治理:eureka分布式配置:config客户端调用工具:rest/feign客户 rpc远程调用说明:阿里巴巴、腾讯、百度注意:大家如果去一些比较大型的互联网公司中,整个公司内部实现rpc通讯的框架、服务帮助治理都是内部自己研发。rpc远程调用框架:HTTPclent、dubbo、feign、grpc、基于netty手写rpcspring cloud一代和二代的区别spring cloud第一代实际上采用Netflix开源的组件整合微服务解决方案。spring cloud第二代实际上就是自己研发和国内的优秀的服务解决微服务框架进行组合。nacos实现服务注册于发现微服务服务治理核心概念Nacos产生背景rpc远程调用框架:HTTP client、grpc、dubbo、rest、openfeign等。传统rpc远程调用中存在哪些问题?超时、安全、服务与服务之间的url地址管理在我们的微服务架构通讯中,服务间依赖非常大,如果使用传统方式管理我们的服务,非常麻烦,所以采用url治理技术,可以实现我们整个实现动态服务注册与发现、本地负载均衡、容错等nacos分布式注册与发现功能 | 分布式配置中心产生于rpc远程调用中,服务的url的治理传统服务注册中心的实现方式把每个服务器的地址信息和端口号人工存放到数据库表中IdserviceIP端口号1itycu.cn192.168.1.180802itycu.cn192.168.1.18000维护成本高没有实现动态自动化基于数据库形式实现服务url治理的缺点分布式注册中心实现原理注册中心的作用管理整个微服务url地址可以实现动态感知注册中心:duboo依赖zookeeper、eureka、consul、nacos、redis、数据库nacos与eureka的区别eureka与zookeeper的区别注册中心的原理keyIP端口号服务名称192.168.1.18080生产者启动时,根据这种存储方式注册到微服务注册中心根据以上存储方式的服务名称获取到IP地址和端口号获取到地址后在本地实现rpc远程调用生产者:提供我们接口被其他服务调用消费者:调用接口实现服务服务注册:提供服务接口地址信息存放Naxos基本介绍实现注册中心和分布式配置中心默认账号密码:nacos使用命令模式实现对nacos的注册使用discovertyClient从注册中心获取接口地址使用rest Template实现rpc远程调用纯手写本地负载均衡轮询算法实现线下服务动态感知
-
直播简介:如何实现spring cloud应用无缝接入ServiceStage应用平台?看直播听华为云微服务专家现场手把手教你应用部署上云 直播时间:7.9 19:30-20:30直播参与流程:1. 加小助手好友报名进入直播群2. 与下方体验任务(获码豆换好礼)3. 等待7.9 19:30-20:30直播开始直播链接:https://bbs.huaweicloud.com/signup/b86fe50c0f7e48b4a8aba2de41f04f8e 奖励&规则任务一:分享任务规则&奖励:点击领取任务(加超链接)活动时间:2020年7月01日-2020年7月09日分享海报,可获得500码豆激励!第一步: 分享以下文案+海报至朋友圈或100人以上技术群(微信、QQ、钉钉不限)。分享可获得500码豆!码豆可用于兑换DevCloud会员中心精美实物礼品。(1)文案:我正在参与华为云直播《对接SpringCloud?一课接入华为云ServiceStage应用平台!》,看华为云专家教你如何实现SpringCloud原生应用快速接入搞性能网关,对接ServiceStage服务平台,看直播还能赢好礼,快来参加吧!(2)海报:添加小助手微信(little-one-1806)或扫下方二维码,回复“分享海报”获取海报和文案。第二步: 分享完成后,截图并点击链接https://www.wjx.top/jq/83359544.aspx上传分享截图;符合要求的截图即可算作分享成功,获得500码豆!码豆奖励会在直播活动结束后15个工作日内发放注意:为保证码豆成功发放,请先登录一次会员中心 任务二:体验任务规则&奖励:活动时间:2020年7月01日-2020年7月09日完成体验产品任务,可获得1000码豆激励!第一步: 体验产品,跟着教程指引点点点就可获得1000码豆!产品体验指导https://bbs.huaweicloud.com/forum/thread-63353-1-1.html码豆可用于兑换DevCloud会员中心精美实物礼品。第二步:体验完成后,截图并在教程帖盖楼回复上传分享截图;符合要求的截图即可算作分享成功,获得1000码豆!码豆奖励会在直播活动结束后15个工作日内发放注意:为保证码豆成功发放,请先登录一次会员中心请务必按照以下格式要求进行回帖打卡,否则无法计算奖励:华为云账号名:XXX(即右上角的字母数字组合ID)微信昵称:XXX 打卡样例图:添加小助手微信:little-one-1806 或 扫下方二维码添加小助手回复”微服务“报名 进群了解更多微服务信息,直播开始前,小助手会在群里pick大家看直播哦~任务三:直播抽奖观看直播并参与评论互动,就有机会获得《微服务架构与实践(第2版)》书籍1本,直播结束后5个工作日内寄出奖品
-
【7.9直播活动码豆发放公示】感谢各位小伙伴参与本期活动!本期直播活动已于7月9日24:00结束,满足码豆发放条件的名单已公布,详见附件表格为保证码豆成功发放,请名单中的小伙伴于7月23日先登录一次会员中心,以便码豆发放码豆将于7月24日发放到账,请及时参与。未及时登录会员中心或未到账的用户请于7月30日登录一次会员中心,我们会通知相关人员于7月31日补发码豆,请及时参与,错过不再补发。直播简介:如何实现spring cloud应用无缝接入ServiceStage应用平台?看直播听华为云微服务专家现场手把手教你应用部署上云 直播时间:7.9 19:30-20:30直播参与流程:1. 加小助手好友报名进入直播群2. 与下方体验任务(获码豆换好礼)3. 等待7.9 19:30-20:30直播开始直播观看链接:https://bbs.huaweicloud.com/signup/b86fe50c0f7e48b4a8aba2de41f04f8e返回活动帖入口:https://bbs.huaweicloud.com/forum/thread-63361-1-1.html体验任务教程动动手指赢码豆,点点点教程开始啦~ 单击体验链接并登录后,进入如下界面: 单击“开始教程”开始体验 跟随指引单击下一步 在应用运维这一步截图并单击完成,结束体验。截图注意一起截取用户名。 完成后,截图并在本帖盖楼回复上传体验截图;前100名符合要求的截图即可算作体验成功,获得1000码豆!码豆奖励会在直播活动结束后15个工作日内发放,请注意查收哦~请务必按照以下格式要求进行回帖打卡,否则无法计算奖励:华为云账号名:XXX(即右上角的字母数字组合ID)微信昵称:XXX 打卡样例图:
-
spring cloud 项目 迁移到 CSE 框架。配置中心的那些yaml 文件,如果迁移到 CSE 配置中心呢? CSE 全局配置项,跟spring cloud 配置中心的 数据如何迁移 让 项目通用的配置抽取出来!
-
加上EnableBinding注解后,启动报端口占用错误猜测是先启动了vertx rest的端口后面再启动spring的tomcat的时候报错了下面是pom和父pom4.0.0com.huawei.aciotparent1.0-SNAPSHOTdevice-common0.0.1-SNAPSHOTdevice-commonDemo project for Spring Boot1.8org.projectlomboklomboktrueorg.apache.servicecombhandler-tracing-zipkinorg.springframework.cloudspring-cloud-stream-test-supporttestorg.springframework.cloudspring-cloud-streamorg.springframework.cloudspring-cloud-stream-binder-kafkaorg.springframework.kafkaspring-kafkacom.huawei.eciotcore-util0.0.1-SNAPSHOT父pomorg.springframework.bootspring-boot-starter-parent2.1.6.RELEASEpomimportorg.springframeworkspring-core5.1.8.RELEASEpomimportcom.huawei.paas.csecse-dependency-spring-boot26.5.62pomimportcom.huawei.paas.csecse-solution-service-engineorg.slf4jslf4j-log4j12org.apache.servicecombspring-boot2-starter-servletorg.springframework.bootspring-boot-starter-testtestyaml: : : device-common : : 8090 : : /device-north : ac-iot : : device : 0.0.1 : : true : development : : : : false : : 192.188.240.32:${server.port} : : /device : : : http://192.88.16.46:30100 : : : : : loadbalance,tracing-consumer : : tracing-provider : : : http://localhost:9411大佬们帮忙看一下是哪里配错了吗stephen_meng 发表于2020-01-06 19:29:01 2020-01-06 19:29:01 最后回复 stephen_meng 2020-01-06 19:30:001863 1
-
微服务架构是一个分布式架构,微服务系统按业务划分服务单元,一个微服务系统往往有很多个服务单元。由于服务单元数量众多,业务的复杂性较高,如果出现了错误和异常,很难去定位。主要体现在一个请求可能需要调用很多个服务,而内部服务的调用复杂性决定了问题难以定位。所以在微服务架构中,必须实现分布式链路追踪,去跟进一个请求到底有哪些服务参与,参与的顺序又是怎样的,从而达到每个请求的步骤清晰可见,出了问题能够快速定位的目的。 在微服务系统中,一个来自用户的请求先到达前端A(如前端界面),然后通过远程调用,到达系统的中间件B、C(如负载均衡、网关等),最后到达后端服务D、E,后端经过一系列的业务逻辑计算,最后将数据返回给用户。对于这样一个请求,经历了这么多个服务,怎么样将它的请求过程用数据记录下来呢?这就需要用到服务链路追踪。Spring Cloud SleuthSpring Cloud Sleuth 为服务之间调用提供链路追踪。通过Sleuth 可以很清楚的了解到一个服务请求经过了哪些服务,每个服务处理花费了多长。从而让我们可以很方便的理清各微服务间的调用关系。此外Sleuth 可以帮助我们:§ 耗时分析: 通过Sleuth 可以很方便的了解到每个采样请求的耗时,从而分析出哪些服务调用比较耗时;§ 可视化错误: 对于程序未捕捉的异常,可以通过集成Zipkin 服务界面上看到;§ 链路优化: 对于调用比较频繁的服务,可以针对这些服务实施一些优化措施。Google开源了Dapper链路追踪组件,并在2010年发表了论文《Dapper, a Large-Scale Distributed Systems Tracing Infrastructure》,这篇论文是业内实现链路追踪的标杆和理论基础,具有很高的参考价值。Spring Cloud Sleuth采用了Google的开源项目Dapper的专业术语。§ Span:基本工作单元,发送一个远程调度任务就会产生一个Span,Span是用一个64位ID唯一标识的,Trace是用另一个64位ID唯一标识的。Span还包含了其他的信息,例如摘要、时间戳事件、Span的ID以及进程ID。§ Trace:由一系列Span组成的,呈树状结构。请求一个微服务系统的API接口,这个API接口需要调用多个微服务单元,调用每个微服务单元都会产生一个新的Span,所有由这个请求产生的Span组成了这个Trace。§ Annotation:用于记录一个事件,一些核心注解用于定义一个请求的开始和结束,这些注解如下。o cs-Client Sent:客户端发送一个请求,这个注解描述了Span的开始。o sr-Server Received:服务端获得请求并准备开始处理它,如果将其sr减去cs时间戳,便可得到网络传输的时间。o ss-Server Sent:服务端发送响应,该注解表明请求处理的完成(当请求返回客户端),用ss的时间戳减去sr时间戳,便可以得到服务器请求的时间。o cr-Client Received:客户端接收响应,此时Span结束,如果cr的时间戳减去cs时间戳,便可以得到整个请求所消耗的时间。Spring Cloud Sleuth 也为我们提供了一套完整的链路解决方案,Spring Cloud Sleuth 可以结合Zipkin,将信息发送到Zipkin,利用Zipkin 的存储来存储链路信息,利用Zipkin UI 来展示数据。ZipkinZipkin是一种分布式链路追踪系统。它有助于收集解决微服务架构中的延迟问题所需的时序数据。它管理这些数据的收集和查找。Zipkin的设计基于Google Dapper论文。跟踪器存在于应用程序中,记录请求调用的时间和元数据。跟踪器使用库,它们的使用对用户是无感知的。例如,Web服务器会在收到请求时和发送响应时会记录相应的时间和一些元数据。一次完整链路请求所收集的数据被称为Span。我们可以使用它来收集各个服务器上请求链路的跟踪数据,并通过它提供的REST API 接口来辅助我们查询跟踪数据以实现对分布式系统的监控程序,从而及时地发现系统中出现的延迟升高问题并找出系统性能瓶颈的根源。除了面向开发的API 接口之外,它也提供了方便的UI 组件来帮助我们直观的搜索跟踪信息和分析请求链路明细,比如:可以查询某段时间内各用户请求的处理时间等。Zipkin 提供了可插拔数据存储方式:In-Memory、MySql、Cassandra 以及Elasticsearch。接下来的测试为方便直接采用In-Memory 方式进行存储,生产推荐Elasticsearch.上图展示了Zipkin 的基础架构,它主要由4 个核心组件构成:§ Collector:收集器组件,它主要用于处理从外部系统发送过来的跟踪信息,将这些信息转换为Zipkin 内部处理的Span 格式,以支持后续的存储、分析、展示等功能。§ Storage:存储组件,它主要对处理收集器接收到的跟踪信息,默认会将这些信息存储在内存中,我们也可以修改此存储策略,通过使用其他存储组件将跟踪信息存储到数据库中。§ RESTful API:API 组件,它主要用来提供外部访问接口。比如给客户端展示跟踪信息,或是外接系统访问以实现监控等。§ Web UI:UI 组件,基于API 组件实现的上层应用。通过UI 组件用户可以方便而有直观地查询和分析跟踪信息。案例实战在本案例一共有三个应用,分别为注册中心,eureka-server、eureka-client、eureka-client-feign,三个应用的基本信息如下:应用名端口作用eureka-server8761注册中心eureka-client8763服务提供者eureka-client-feign8765服务消费者其中eureka-server 应用为注册中心,其他两个应用向它注册。eureka-client为服务提供者,提供了一个RESTAPI,eureka-client-feign为服务消费者,通过Feign Client向服务提供者消费服务。在之前的文章已经讲述了如何如何搭建服务注册中心,在这里就省略这一部分内容。服务提供者提供一个REST接口,服务消费者通过FeignClient消费服务。服务提供者eureka-client服务提供者,对外提供一个RESTAPI,并向服务注册中心注册,这部分内容,不再讲述,见源码。需要在工程的pom文件加上sleuth的起步依赖和zipkin的起步依赖,代码如下:1. <dependency>2. <groupId>org.springframework.cloud</groupId>3. <artifactId>spring-cloud-starter-sleuth</artifactId>4. </dependency>5. <dependency>6. <groupId>org.springframework.cloud</groupId>7. <artifactId>spring-cloud-starter-zipkin</artifactId>8. </dependency>在工程的配置文件application.yml需要做以下的配置:1. spring:2. sleuth:3. web:4. client:5. enabled:true6. sampler:7. probability:1.0 # 将采样比例设置为1.0,也就是全部都需要。默认是0.18. zipkin:9. base-url:http://localhost:9411/ # 指定了Zipkin 服务器的地址其中spring.sleuth.web.client.enable为true设置的是web开启sleuth功能;spring.sleuth.sampler.probability可以设置为小数,最大值为1.0,当设置为1.0时就是链路数据100%收集到zipkin-server,当设置为0.1时,即10%概率收集链路数据;spring.zipkin.base-url设置zipkin-server的地址。对外提供一个Api,代码如下:1. @RestController2. publicclassHiController{3. 4. @Value("${server.port}")5. Stringport;6. @GetMapping("/hi")7. publicStringhome(@RequestParamStringname){8. return"hi "+name+",i am from port:"+port;9. }10. 11. }服务消费者服务消费者通过FeignClient消费服务提供者提供的服务。同服务提供者一样,需要在工程的pom文件加上sleuth的起步依赖和zipkin的起步依赖,另外也需要在配置文件application.yml做相关的配置,具体同服务提供者。服务消费者通过feignClient进行服务消费,feignclient代码如下:1. @FeignClient(value ="eureka-client",configuration =FeignConfig.class)2. publicinterfaceEurekaClientFeign{3. 4. @GetMapping(value ="/hi")5. StringsayHiFromClientEureka(@RequestParam(value ="name")Stringname);6. }servcie层代码如下:1. @Service2. publicclassHiService{3. 4. @Autowired5. EurekaClientFeigneurekaClientFeign;6. 7. 8. publicStringsayHi(Stringname){9. return eurekaClientFeign.sayHiFromClientEureka(name);10. }11. }controller代码如下:1. @RestController2. publicclassHiController{3. @Autowired4. HiServicehiService;5. 6. @GetMapping("/hi")7. publicStringsayHi(@RequestParam(defaultValue ="forezp",required =false)Stringname){8. returnhiService.sayHi(name);9. }上面的代码对外暴露一个API,通过FeignClient的方式调用eureka-client的服务。zipkin-server在Spring Cloud D版本,zipkin-server通过引入依赖的方式构建工程,自从E版本之后,这一方式改变了,采用官方的jar形式启动,所以需要通过下载官方的jar来启动,也通过以下命令一键启动:1. curl -sSL https://zipkin.io/quickstart.sh | bash -s2. java -jar zipkin.jar上面的第一行命令会从zipkin官网下载官方的jar包。如果是window系统,建议使用gitbash执行上面的命令。如果用Docker 的话,使用以下命令:1. docker run -d -p 9411:9411openzipkin/zipkin通过java -jar zipkin.jar的方式启动之后,在浏览器**问lcoalhost:9411,显示的界面如下:链路数据验证依次启动eureka-server,eureka-client,eureka-client-feign的三个应用,等所有应用启动完成后,在浏览器**问http://localhost:8765/hi(如果报错,是服务与发现需要一定的时间,耐心等待几十秒),访问成功后,再次在浏览器**问zipkin-server的页面,显示如下:从上图可以看出每次请求所消耗的时间,以及一些span的信息。从上图可以看出具体的服务依赖关系,eureka-feign-client依赖了eureka-client。使用rabbitmq进行链路数据收集在上面的案例中使用的http请求的方式将链路数据发送给zipkin-server,其实还可以使用rabbitmq的方式进行服务的消费。使用rabbitmq需要安装rabbitmq程序,下载地址http://www.rabbitmq.com/。下载完成后,需要eureka-client和eureka-client-feign的起步依赖加上rabbitmq的依赖,依赖如下:1. <dependency>2. <groupId>org.springframework.cloud</groupId>3. <artifactId>spring-cloud-stream-binder-rabbit</artifactId>4. </dependency>在配置文件上需要配置rabbitmq的配置,配置信息如下:1. spring:2. rabbitmq:3. host:localhost4. username:guest5. password:guest6. port:5672另外需要把spring.zipkin.base-url去掉。在上面2个工程中,rabbitmq通过发送链路数据,那么zipkin-server是怎么样知道rabbitmq的地址呢,怎么监听收到的链路数据呢?这需要在程序启动的时候,通过环境变量的形式到环境中,然后zikin-server从环境变量中读取。可配置的属性如下:属性环境变量描述zipkin.collector.rabbitmq.addressesRABBIT_ADDRESSES用逗号分隔的RabbitMQ 地址列表,例如localhost:5672,localhost:5673zipkin.collector.rabbitmq.passwordRABBIT_PASSWORD连接到RabbitMQ 时使用的密码,默认为guestzipkin.collector.rabbitmq.usernameRABBIT_USER连接到RabbitMQ 时使用的用户名,默认为guestzipkin.collector.rabbitmq.virtual-hostRABBITVIRTUALHOST使用的RabbitMQ virtual host,默认为/zipkin.collector.rabbitmq.use-sslRABBITUSESSL设置为true则用SSL 的方式与RabbitMQ 建立链接zipkin.collector.rabbitmq.concurrencyRABBIT_CONCURRENCY并发消费者数量,默认为1zipkin.collector.rabbitmq.connection-timeoutRABBITCONNECTIONTIMEOUT建立连接时的超时时间,默认为60000毫秒,即1 分钟zipkin.collector.rabbitmq.queueRABBIT_QUEUE从中获取span 信息的队列,默认为zipkin比如,通过以下命令启动:1. RABBIT_ADDRESSES=localhost java-jar zipkin.jar上面的命令等同于一下的命令:1. java -jar zipkin.jar --zipkin.collector.rabbitmq.addressed=localhost用上面的2条命令中的任何一种方式重新启动zipkin-server程序,并重新启动eureka-client、eureka-server、eureka-client-feign,动完成后在浏览器**问http://localhost:8765/hi,再访问http://localhost:9411/zipkin/,就可以看到通过Http方式发送链路数据一样的接口。自定义Tag在页面上可以查看每个请求的traceId,每个trace又包含若干的span,每个span又包含了很多的tag,自定义tag可以通过Tracer这个类来自定义。1. @Autowired2. Tracertracer;3. 4. @GetMapping("/hi")5. publicStringhome(@RequestParamStringname){6. tracer.currentSpan().tag("name","forezp");7. return"hi "+name+",i am from port:"+port;8. }将链路数据存储在Mysql数据库中上面的例子是将链路数据存在内存中,只要zipkin-server重启之后,之前的链路数据全部查找不到了,zipkin是支持将链路数据存储在mysql、cassandra、elasticsearch中的。现在讲解如何将链路数据存储在Mysql数据库中。首先需要初始化zikin存储在Mysql的数据的scheme,可以在这里查看https://github.com/openzipkin/zipkin/blob/master/zipkin-storage/mysql-v1/src/main/resources/mysql.sql,具体如下:1. CREATE TABLE IF NOT EXISTS zipkin_spans (2. `trace_id_high`BIGINT NOT NULL DEFAULT 0COMMENT 'If non zero, this means the trace uses 128 bit traceIds instead of 64 bit',3. `trace_id`BIGINT NOT NULL,4. `id`BIGINT NOT NULL,5. `name`VARCHAR(255)NOT NULL,6. `parent_id`BIGINT,7. `debug`BIT(1),8. `start_ts`BIGINT COMMENT 'Span.timestamp(): epoch micros used for endTs query and to implement TTL',9. `duration`BIGINT COMMENT 'Span.duration(): micros used for minDuration and maxDuration query'10. )ENGINE=InnoDBROW_FORMAT=COMPRESSED CHARACTER SET=utf8 COLLATE utf8_general_ci;11. 12. ALTER TABLE zipkin_spans ADD UNIQUE KEY(`trace_id_high`,`trace_id`,`id`)COMMENT 'ignore insert on duplicate';13. ALTER TABLE zipkin_spans ADD INDEX(`trace_id_high`,`trace_id`,`id`)COMMENT 'for joining with zipkin_annotations';14. ALTER TABLE zipkin_spans ADD INDEX(`trace_id_high`,`trace_id`)COMMENT 'for getTracesByIds';15. ALTER TABLE zipkin_spans ADD INDEX(`name`)COMMENT 'for getTraces and getSpanNames';16. ALTER TABLE zipkin_spans ADD INDEX(`start_ts`)COMMENT 'for getTraces ordering and range';17. 18. CREATE TABLE IF NOT EXISTS zipkin_annotations (19. `trace_id_high`BIGINT NOT NULL DEFAULT 0COMMENT 'If non zero, this means the trace uses 128 bit traceIds instead of 64 bit',20. `trace_id`BIGINT NOT NULL COMMENT 'coincides with zipkin_spans.trace_id',21. `span_id`BIGINT NOT NULL COMMENT 'coincides with zipkin_spans.id',22. `a_key`VARCHAR(255)NOT NULL COMMENT 'BinaryAnnotation.key or Annotation.value if type == -1',23. `a_value`BLOB COMMENT 'BinaryAnnotation.value(), which must be smaller than 64KB',24. `a_type`INT NOT NULL COMMENT 'BinaryAnnotation.type() or -1 if Annotation',25. `a_timestamp`BIGINT COMMENT 'Used to implement TTL; Annotation.timestamp or zipkin_spans.timestamp',26. `endpoint_ipv4`INT COMMENT 'Null when Binary/Annotation.endpoint is null',27. `endpoint_ipv6`BINARY(16)COMMENT 'Null when Binary/Annotation.endpoint is null, or no IPv6 address',28. `endpoint_port`SMALLINT COMMENT 'Null when Binary/Annotation.endpoint is null',29. `endpoint_service_name`VARCHAR(255)COMMENT 'Null when Binary/Annotation.endpoint is null'30. )ENGINE=InnoDBROW_FORMAT=COMPRESSED CHARACTER SET=utf8 COLLATE utf8_general_ci;31. 32. ALTER TABLE zipkin_annotations ADD UNIQUE KEY(`trace_id_high`,`trace_id`,`span_id`,`a_key`,`a_timestamp`)COMMENT 'Ignore insert on duplicate';33. ALTER TABLE zipkin_annotations ADD INDEX(`trace_id_high`,`trace_id`,`span_id`)COMMENT 'for joining with zipkin_spans';34. ALTER TABLE zipkin_annotations ADD INDEX(`trace_id_high`,`trace_id`)COMMENT 'for getTraces/ByIds';35. ALTER TABLE zipkin_annotations ADD INDEX(`endpoint_service_name`)COMMENT 'for getTraces and getServiceNames';36. ALTER TABLE zipkin_annotations ADD INDEX(`a_type`)COMMENT 'for getTraces and autocomplete values';37. ALTER TABLE zipkin_annotations ADD INDEX(`a_key`)COMMENT 'for getTraces and autocomplete values';38. ALTER TABLE zipkin_annotations ADD INDEX(`trace_id`,`span_id`,`a_key`)COMMENT 'for dependencies job';39. 40. CREATE TABLE IF NOT EXISTS zipkin_dependencies (41. `day`DATE NOT NULL,42. `parent`VARCHAR(255)NOT NULL,43. `child`VARCHAR(255)NOT NULL,44. `call_count`BIGINT,45. `error_count`BIGINT46. )ENGINE=InnoDBROW_FORMAT=COMPRESSED CHARACTER SET=utf8 COLLATE utf8_general_ci;47. 48. ALTER TABLE zipkin_dependencies ADD UNIQUE KEY(`day`,`parent`,`child`);在数据库中初始化上面的脚本之后,需要做的就是zipkin-server如何连接数据库。zipkin如何连数据库同连接rabbitmq一样。zipkin连接数据库的属性所对应的环境变量如下:属性环境变量描述zipkin.torage.typeSTORAGE_TYPE默认的为mem,即为内存,其他可支持的为cassandra、cassandra3、elasticsearch、mysqlzipkin.torage.mysql.hostMYSQL_HOST数据库的host,默认localhostzipkin.torage.mysql.portMYSQLTCPPORT数据库的端口,默认3306zipkin.torage.mysql.usernameMYSQL_USER连接数据库的用户名,默认为空zipkin.torage.mysql.passwordMYSQL_PASS连接数据库的密码,默认为空zipkin.torage.mysql.dbMYSQL_DBzipkin使用的数据库名,默认是zipkinzipkin.torage.mysql.max-activeMYSQLMAXCONNECTIONS最大连接数,默认是101. STORAGE_TYPE=mysql MYSQL_HOST=localhost MYSQL_TCP_PORT=3306MYSQL_USER=root MYSQL_PASS=123456MYSQL_DB=zipkin java -jar zipkin.jar等同于以下的命令1. java -jar zipkin.jar --zipkin.torage.type=mysql --zipkin.torage.mysql.host=localhost --zipkin.torage.mysql.port=3306--zipkin.torage.mysql.username=root --zipkin.torage.mysql.password=123456使用上面的命令启动zipkin.jar工程,然后再浏览数**问http://localhost:8765/hi,再访问http://localhost:9411/zipkin/,可以看到链路数据。这时去数据库查看数据,也是可以看到存储在数据库的链路数据,如下:这时重启应用zipkin.jar,再次在浏览器**问http://localhost:9411/zipkin/,仍然可以得到之前的结果,证明链路数据存储在数据库中,而不是内存中。将链路数据存在在Elasticsearch中zipkin-server支持将链路数据存储在ElasticSearch中。读者需要自行安装ElasticSearch和Kibana,下载地址为https://www. elastic.co/products/elasticsearch。安装完成后启动,其中ElasticSearch的默认端口号为9200,Kibana的默认端口号为5601。同理,zipkin连接elasticsearch也是从环境变量中读取的,elasticsearch相关的环境变量和对应的属性如下:属性环境变量描述zipkin.torage.elasticsearch.hostsES_HOSTSES_HOSTS,默认为空zipkin.torage.elasticsearch.pipelineES_PIPELINEES_PIPELINE,默认为空zipkin.torage.elasticsearch.max-requestsESMAXREQUESTSESMAXREQUESTS,默认为64zipkin.torage.elasticsearch.timeoutES_TIMEOUTES_TIMEOUT,默认为10szipkin.torage.elasticsearch.indexES_INDEXES_INDEX,默认是zipkinzipkin.torage.elasticsearch.date-separatorESDATESEPARATORESDATESEPARATOR,默认为“-”zipkin.torage.elasticsearch.index-shardsESINDEXSHARDSESINDEXSHARDS,默认是5zipkin.torage.elasticsearch.index-replicasESINDEXREPLICASESINDEXREPLICAS,默认是1zipkin.torage.elasticsearch.usernameES_USERNAMEES的用户名,默认为空zipkin.torage.elasticsearch.passwordES_PASSWORDES的密码,默认是为空采用以下命令启动zipkin-server:1. STORAGE_TYPE=elasticsearch ES_HOSTS=http://localhost:9200 ES_INDEX=zipkin java -jar zipkin.jar1. java -jar zipkin.jar --STORAGE_TYPE=elasticsearch --ES_HOSTS=http://localhost:9200 --ES_INDEX=zipkin 1. java -jar zipkin.jar --STORAGE_TYPE=elasticsearch --ES_HOSTS=http://localhost:9200 --ES_INDEX=zipkin 1. java -jar zipkin.jar --zipkin.torage.type=elasticsearch --zipkin.torage.elasticsearch.hosts=http://localhost:9200 --zipkin.torage.elasticsearch.index=zipkin 启动完成后,然后在浏览数**问http://localhost:8765/hi,再访问http://localhost:9411/zipkin/,可以看到链路数据。这时链路数据存储在ElasticSearch。在zipkin上展示链路数据链路数据存储在ElasticSearch中,ElasticSearch可以和Kibana结合,将链路数据展示在Kibana上。安装完成Kibana后启动,Kibana默认会向本地端口为9200的ElasticSearch读取数据。Kibana默认的端口为5601,访问Kibana的主页http://localhost:5601,其界面如下图所示。在上图的界面中,单击“Management”按钮,然后单击“Add New”,添加一个index。我们将在上节ElasticSearch中写入链路数据的index配置为“zipkin”,那么在界面填写为“zipkin-*”,单击“Create”按钮,界面如下图所示:创建完成index后,单击“Discover”,就可以在界面上展示链路数据了,展示界面如下图所示。参考资料https://zipkin.io/https://github.com/spring-cloud/spring-cloud-sleuthhttps://cloud.spring.io/spring-cloud-static/spring-cloud-sleuth/2.1.0.RELEASE/single/spring-cloud-sleuth.htmlhttps://github.com/openzipkin/zipkin/blob/master/zipkin-server/src/main/resources/zipkin-server-shared.ymlhttps://github.com/openzipkin/zipkin/blob/master/zipkin-storage/mysql-v1/src/main/resources/mysql.sqlhttps://windmt.com/2018/04/24/spring-cloud-12-sleuth-zipkin/https://segmentfault.com/a/119000001569767http://www.cnblogs.com/JreeyQi/p/9336692.html原创作者:方志朋方志朋简介:SpringCloud中国社区联合创始人,博客访问量突破一千万,爱好开源,热爱分享,活跃于各大社区,保持着非常强的学习驱动力,终身学习践行者,终身学习受益者。目前就职于国内某家知名互联网保险公司,担任DEVOPS工程师,对微服务领域和持续集成领域研究较深,精通微服务框架SpringCloud
-
什么是ConsulConsul是HashiCorp公司推出的开源软件,使用GO语言编写,提供了分布式系统的服务注册和发现、配置等功能,这些功能中的每一个都可以根据需要单独使用,也可以一起使用以构建全方位的服务网格。Consul不仅具有服务治理的功能,而且使用分布式一致协议RAFT算法实现,有多数据中心的高可用方案,并且很容易和Spring Cloud等微服务框架集成,使用起来非常的简单,具有简单、易用、可插排等特点。使用简而言之,Consul提供了一种完整的服务网格解决方案。Consul具有以下的特点和功能§ 服务发现:Consul的客户端可以向Consul注册服务,例如api服务或者mysql服务,其他客户端可以使用Consul来发现服务的提供者。Consul支持使用DNS或HTTP来注册和发现服务。§ 运行时健康检查:Consul客户端可以提供任意数量的运行状况检查机制,这些检查机制可以是给定服务(“是Web服务器返回200 OK”)或本地节点(“内存利用率低于90%”)相关联。这些信息可以用来监控群集的运行状况,服务发现组件可以使用这些监控信息来路由流量,可以使流量远离不健康的服务。§ KV存储:应用程序可以将Consul的键/值存储用于任何需求,包括动态配置,功能标记,协调,领导者选举等。它采用HTTP API使其易于使用。§ 安全服务通信:Consul可以为服务生成和分发TLS证书,以建立相互的TLS连接。§ 多数据中心:Consul支持多个数据中心。这意味着Consul的用户不必担心构建额外的抽象层以扩展到多个区域。Consul原理每个提供服务的节点都运行了Consul的代理,运行代理不需要服务发现和获取配置的KV键值对,代理只负责监控检查。代理节点可以和一个或者多个Consul server通讯。Consul服务器是存储和复制数据的地方。服务器本身选出了领导者。虽然Consul可以在一台服务器上运行,但建议使用3到5,以避免导致数据丢失的故障情况。建议为每个数据中心使用一组Consul服务器。如果你的组件需要发现服务,可以查询任何Consul Server或任何Consul客户端,Consul客户端会自动将查询转发给Consul Server。需要发现其他服务或节点的基础架构组件可以查询任何Consul服务器或任何Consul代理。代理会自动将查询转发给服务器。每个数据中心都运行Consul服务器集群。发生跨数据中心服务发现或配置请求时,本地Consul服务器会将请求转发到远程数据中心并返回结果。术语§ Agent agent是一直运行在Consul集群中每个成员上的守护进程。通过运行consul agent 来启动。agent可以运行在client或者server模式。指定节点作为client或者server是非常简单的,除非有其他agent实例。所有的agent都能运行DNS或者HTTP接口,并负责运行时检查和保持服务同步。§ Client 一个Client是一个转发所有RPC到server的代理。这个client是相对无状态的。client唯一执行的后台活动是加入LAN gossip池。这有一个最低的资源开销并且仅消耗少量的网络带宽。§ Server 一个server是一个有一组扩展功能的代理,这些功能包括参与Raft选举,维护集群状态,响应RPC查询,与其他数据中心交互WAN gossip和转发查询给leader或者远程数据中心。§ DataCenter 虽然数据中心的定义是显而易见的,但是有一些细微的细节必须考虑。例如,在EC2中,多个可用区域被认为组成一个数据中心?我们定义数据中心为一个私有的,低延迟和高带宽的一个网络环境。这不包括访问公共网络,但是对于我们而言,同一个EC2中的多个可用区域可以被认为是一个数据中心的一部分。§ Consensus 在我们的文档中,我们使用Consensus来表明就leader选举和事务的顺序达成一致。由于这些事务都被应用到有限状态机上,Consensus暗示复制状态机的一致性。§ Gossip Consul建立在Serf的基础之上,它提供了一个用于多播目的的完整的gossip协议。Serf提供成员关系,故障检测和事件广播。更多的信息在gossip文档中描述。这足以知道gossip使用基于UDP的随机的点到点通信。§ LAN Gossip 它包含所有位于同一个局域网或者数据中心的所有节点。§ WAN Gossip 它只包含Server。这些server主要分布在不同的数据中心并且通常通过因特网或者广域网通信。§ RPC 远程过程调用。这是一个允许client请求server的请求/响应机制。 让我们分解这张图并描述每个部分。首先,我们能看到有两个数据中心,标记为“1”和“2”。Consul对多数据中心有一流的支持并且希望这是一个常见的情况。在每个数据中心,client和server是混合的。一般建议有3-5台server。这是基于有故障情况下的可用性和性能之间的权衡结果,因为越多的机器加入达成共识越慢。然而,并不限制client的数量,它们可以很容易的扩展到数千或者数万台。同一个数据中心的所有节点都必须加入gossip协议。这意味着gossip协议包含一个给定数据中心的所有节点。这服务于几个目的:第一,不需要在client上配置server地址。发现都是自动完成的。第二,检测节点故障的工作不是放在server上,而是分布式的。这是的故障检测相比心跳机制有更高的可扩展性。第三:它用来作为一个消息层来通知事件,比如leader选举发生时。每个数据中心的server都是Raft节点集合的一部分。这意味着它们一起工作并选出一个leader,一个有额外工作的server。leader负责处理所有的查询和事务。作为一致性协议的一部分,事务也必须被复制到所有其他的节点。因为这一要求,当一个非leader得server收到一个RPC请求时,它将请求转发给集群leader。server节点也作为WAN gossip Pool的一部分。这个Pool不同于LAN Pool,因为它是为了优化互联网更高的延迟,并且它只包含其他Consul server节点。这个Pool的目的是为了允许数据中心能够以low-touch的方式发现彼此。这使得一个新的数据中心可以很容易的加入现存的WAN gossip。因为server都运行在这个pool中,它也支持跨数据中心请求。当一个server收到来自另一个数据中心的请求时,它随即转发给正确数据中想一个server。该server再转发给本地leader。这使得数据中心之间只有一个很低的耦合,但是由于故障检测,连接缓存和复用,跨数据中心的请求都是相对快速和可靠的。Consul 服务注册发现流程Consul在业界最广泛的用途就是作为服务注册中心,同Eureka类型,consul作为服务注册中心,它的注册和发现过程如下图:在上面的流程图上有三个角色,分别为服务注册中心、服务提供者、服务消费者。§ 服务提供者Provider启动的时候,会向Consul发送一个请求,将自己的host、ip、应用名、健康检查等元数据信息发送给Consul§ Consul 接收到Provider 的注册后,定期向Provider 发送健康检查的请求,检验Provider是否健康§ 服务消费者Consumer会从注册中心Consul中获取服务注册列表,当服务消费者消费服务时,根据应用名从服务注册列表获取到具体服务的实例(1个或者多个),从而完成服务的调用。Consul VS EurekaEureka是一种服务发现工具。该体系结构主要是客户端/服务器,每个数据中心有一组Eureka服务器,通常每个可用区域一个。通常,Eureka的客户使用嵌入式SDK来注册和发现服务。对于非本地集成的客户端,使用Ribbon等边车通过Eureka透明地发现服务。Eureka使用尽力而为的复制提供弱一致的服务视图。当客户端向服务器注册时,该服务器将尝试复制到其他服务器但不提供保证。服务注册的生存时间很短(TTL),要求客户端对服务器进行心跳检测。不健康的服务或节点将停止心跳,导致它们超时并从注册表中删除。发现请求可以路由到任何服务,由于尽力复制,这些服务可以提供过时或丢失的数据。这种简化的模型允许轻松的集群管理和高可扩展性。Consul提供了一系列超级功能,包括更丰富的运行状况检查,键/值存储和多数据中心感知。Consul需要每个数据中心中的一组服务器,以及每个客户端上的代理,类似于使用像Ribbon这样的边车。Consul代理允许大多数应用程序不知道Consul,通过配置文件执行服务注册以及通过DNS或负载平衡器sidecars进行发现。Consul提供强大的一致性保证,因为服务器使用Raft协议复制状态。Consul支持丰富的运行状况检查,包括TCP,HTTP,Nagios / Sensu兼容脚本或基于的Eureka的TTL。客户端节点参与基于gossip的健康检查,该检查分发健康检查的工作,而不像集中式心跳,这成为可扩展性挑战。发现请求被路由到当选的Consul领导者,这使他们默认情况下非常一致。允许过时读取的客户端允许任何服务器处理其请求,从而允许像Eureka一样的线性可伸缩性。Consul的强烈一致性意味着它可以用作领导者选举和集群协调的锁定服务。Eureka不提供类似的保证,并且通常需要为需要执行协调或具有更强一致性需求的服务运行ZooKeeper。Consul提供了支持面向服务的体系结构所需的功能工具包。这包括服务发现,还包括丰富的运行状况检查,锁定,键/值,多数据中心联合,事件系统和ACL。Consul和consul-template和envconsul等工具生态系统都试图最大限度地减少集成所需的应用程序更改,以避免需要通过SDK进行本机集成。Eureka是更大的Netflix OSS套件的一部分,该套件期望应用程序相对同质且紧密集成。因此,Eureka只解决了有限的一部分问题,期望其他工具如ZooKeeper可以同时使用。Eureka Server端采用的是P2P的复制模式,但是它不保证复制操作一定能成功,因此它提供的是一个最终一致性的服务实例视图;Client端在Server端的注册信息有一个带期限的租约,一旦Server端在指定期间没有收到Client端发送的心跳,则Server端会认定为Client端注册的服务是不健康的,定时任务将会将其从注册表中删除。Consul与Eureka不同,Consul采用Raft算法,可以提供强一致性的保证,Consul的agent相当于Netflix Ribbon + Netflix Eureka Client,而且对应用来说相对透明,同时相对于Eureka这种集中式的心跳检测机制,Consul的agent可以参与到基于goosip协议的健康检查,分散了server端的心跳检测压力。除此之外,Consul为多数据中心提供了开箱即用的原生支持等。Consul下载和安装Consul采用Go语言编写,支持Linux、Mac、Windows等各大操作系统,本文使用windows操作系统,下载地址:https://www.consul.io/downloads.html,下完成后解压到计算机目录下,解压成功后,只有一个可执行的consul.exe可执行文件。打开cmd终端,切换到目录,执行以下命令:1. consul --version终端显示如下:1. Consulv1.4.22. Protocol2spoken bydefault, understands 2to 3(agent will automatically usep3. rotocol >2whenspeaking to compatible agents)证明consul下载成功了,并可执行。consul的一些常见的执行命令如下:命令解释示例agent运行一个consul agentconsul agent -devjoin将agent加入到consul集群consul join IPmembers列出consul cluster集群中的membersconsul membersleave将节点移除所在集群consul leave更多命令请查看官方网站:https://www.consul.io/docs/commands/index.html开发模式启动:1. consul agent -dev 启动成功,在浏览器**问:http://localhost:8500,显示的界面如下:spring cloud consul该项目通过自动配置并绑定到Spring环境和其他Spring编程模型成语,为Spring Boot应用程序提供Consul集成。通过几个简单的注释,您可以快速启用和配置应用程序中的常见模式,并使用基于Consul的组件构建大型分布式系统。提供的模式包括服务发现,控制总线和配置。智能路由(Zuul)和客户端负载平衡(Ribbon),断路器(Hystrix)通过与Spring Cloud Netflix的集成提供。使用spring cloud consul来服务注册与发现本小节以案例的形式来讲解如何使用Spring Cloud Consul来进行服务注册和发现的,并且使用Feign来消费服务。再讲解之前,已经启动consul的agent,并且在浏览器上http://localhost:8500能够显示正确的页面。本案例一共有2个工程,分别如下:工程名端口描述consul-provider8763服务提供者consul-consumer8765服务消费者其中,服务提供者和服务消费者分别向consul注册,注册完成后,服务消费者通过FeignClient来消费服务提供者的服务。服务提供者consul-provider创建一个工程consul-provider,在工程的pom文件引入以下依赖,包括consul-discovery的起步依赖,该依赖是spring cloud consul用来向consul 注册和发现服务的依赖,采用REST API的方式进行通讯。另外加上web的起步依赖,用于对外提供REST API。代码如下:1. <dependency>2. <groupId>org.springframework.cloud</groupId>3. <artifactId>spring-cloud-starter-consul-discovery</artifactId>4. </dependency>5. <dependency>6. <groupId>org.springframework.boot</groupId>7. <artifactId>spring-boot-starter-web</artifactId>8. </dependency>在工程的配置文件application.yml做下以下配置:1. server:2. port: 87633. spring:4. application:5. name: consul-provider6. cloud:7. consul:8. host: localhost9. port: 850010. discovery:11. serviceName: consul-provider上面的配置,指定了程序的启动端口为8763,应用名为consul-provider,consul注册中心的地址为localhost:8500在程序员的启动类ConsulProviderApplication加上@EnableDiscoveryClient注解,开启服务发现的功能。1. @SpringBootApplication2. @EnableDiscoveryClient3. publicclassConsulProviderApplication{4. 5. publicstaticvoidmain(String[] args) {6. SpringApplication.run(ConsulProviderApplication.class, args);7. }8. 9. }写一个RESTAPI,该API为一个GET请求,返回当前程序的启动端口,代码如下。1. @RestController2. publicclassHiController{3. 4. @Value("${server.port}")5. Stringport;6. @GetMapping("/hi")7. publicStringhome(@RequestParamStringname) {8. return"hi "+name+",i am from port:"+port;9. }10. 11. }启动工程,在浏览器**问http://localhost:8500,页面显示如下:从上图可知,consul-provider服务已经成功注册到consul上面去了。服务消费者consul-provider服务消费者的搭建过程同服务提供者,在pom文件中引入的依赖同服务提供者,在配置文件application.yml配置同服务提供者,不同的点在端口为8765,服务名为consul-consumer。写一个FeignClient,该FeignClient调用consul-provider的REST API,代码如下:1. @FeignClient(value = "consul-provider")2. publicinterfaceEurekaClientFeign{3. 4. 5. @GetMapping(value = "/hi")6. StringsayHiFromClientEureka(@RequestParam(value = "name") Stringname);7. }Service层代码如下:1. @Service2. publicclassHiService{3. 4. @Autowired5. EurekaClientFeigneurekaClientFeign;6. 7. 8. publicStringsayHi(Stringname){9. return eurekaClientFeign.sayHiFromClientEureka(name);10. }11. }对外提供一个REST API,该API调用了consul-provider的服务,代码如下:1. @RestController2. publicclassHiController{3. @Autowired4. HiServicehiService;5. 6. @GetMapping("/hi")7. publicStringsayHi(@RequestParam( defaultValue = "forezp",required = false)Stringname){8. returnhiService.sayHi(name);9. }10. }在浏览器**问http://localhost:8765/hi,浏览器响应如下:hi forezp,i am from port:8763这说明consul-consumer已经成功调用了consul-provider的服务。这说明consul-provider的服务已经注册到了consul的注册中心上面去了。consul-consumer能够获取注册中心的注册列表来获来消费服务。使用Spring Cloud Consul Config来做服务配置中心Consul不仅能用来服务注册和发现,Consul而且支持Key/Value键值对的存储,可以用来做配置中心。Spring Cloud 提供了Spring Cloud Consul Config依赖去和Consul相集成,用来做配置中心。现在以案例的形式来讲解如何使用Consul作为配置中心,本案例在上一个案例的consul-provider基础上进行改造。首先在工程的pom文件加上consul-config的起步依赖,代码如下:1. <dependency>2. <groupId>org.springframework.cloud</groupId>3. <artifactId>spring-cloud-starter-consul-config</artifactId>4. </dependency>然后在配置文件application.yml加上以下的以下的配置,配置如下:1. spring:2. profiles:3. active: dev 上面的配置指定了SpringBoot启动时的读取的profiles为dev。然后再工程的启动配置文件bootstrap.yml文件中配置以下的配置:1. spring:2. application:3. name: consul-provider4. cloud:5. consul:6. host: localhost7. port: 85008. discovery:9. serviceName: consul-provider10. config:11. enabled: true12. format: yaml 13. prefix: config 14. profile-separator: ':' 15. data-key: data 关于spring.cloud.consul.config的配置项描述如下:§ enabled 设置config是否启用,默认为true§ format 设置配置的值的格式,可以yaml和properties§ prefix 设置配的基本目录,比如config§ defaultContext 设置默认的配置,被所有的应用读取,本例子没用的§ profileSeparator profiles配置分隔符,默认为‘,’§ date-key为应用配置的key名字,值为整个应用配置的字符串。网页**问consul的KV存储的管理界面,即http://localhost:8500/ui/dc1/kv,创建一条记录,key值为:config/consul-provider:dev/data value值如下:1. foo:2. bar: bar13. server:4. port: 8081在consul-provider工程新建一个API,该API返回从consul 配置中心读取foo.bar的值,代码如下:1. @RestController2. publicclassFooBarController{3. 4. @Value("${foo.bar}")5. StringfooBar;6. 7. @GetMapping("/foo")8. publicStringgetFooBar() {9. returnfooBar;10. }11. }启动工程,可以看到程序的启动端口为8081,即是consul的配置中心配置的server.port端口。工程启动完成后,在浏览器**问http://localhost:8081/foo,页面显示bar1。由此可知,应用consul-provider已经成功从consul的配置中心读取了配置foo.bar的配置。动态刷新配置当使用spring cloud config作为配置中心的时候,可以使用spring cloud config bus支持动态刷新配置。Spring Cloud Comsul Config默认就支持动态刷新,只需要在需要动态刷新的类上加上@RefreshScope注解即可,修改代码如下:1. @RestController2. @RefreshScope3. publicclassFooBarController{4. 5. @Value("${foo.bar}")6. StringfooBar;7. 8. @GetMapping("/foo")9. publicStringgetFooBar() {10. returnfooBar;11. }12. }启动consul-provider工程,在浏览器**问http://localhost:8081/foo,页面显示bar1。然后在网页**问consul的KV存储的管理界面,即http://localhost:8500/ui/dc1/kv,修改config/consul-provider:dev/data的值,修改后的值如下:1. foo:2. bar: bar23. server: 4. port: 8081此时不重新启动consul-provider,在浏览器**问http://localhost:8081/foo,页面显示bar2。可见foo.bar的最新配置在应用不重启的情况下已经生效。注意事项§ consul支持的KV存储的Value值不能超过512KB§ Consul的dev模式,所有数据都存储在内存中,重启Consul的时候会导致所有数据丢失,在正式的环境中,Consul的数据会持久化,数据不会丢失。参考资料https://www.consul.io/intro/index.htmlhttps://www.consul.io/docs/internals/architecture.htmlhttps://www.consul.io/intro/vs/eureka.htmlhttp://www.ityouknow.com/springcloud/2018/07/20/spring-cloud-consul.htmlhttps://springcloud.cc/spring-cloud-consul.htmlhttps://www.cnblogs.com/lsf90/p/6021465.htmlhttps://blog.csdn.net/longgeqiaojie304/article/details/85227936 原创作者:方志朋方志朋简介:SpringCloud中国社区联合创始人,博客访问量突破一千万,爱好开源,热爱分享,活跃于各大社区,保持着非常强的学习驱动力,终身学习践行者,终身学习受益者。目前就职于国内某家知名互联网保险公司,担任DEVOPS工程师,对微服务领域和持续集成领域研究较深,精通微服务框架SpringCloud
-
原创作者:方志朋方志朋简介:SpringCloud中国社区联合创始人,博客访问量突破一千万,爱好开源,热爱分享,活跃于各大社区,保持着非常强的学习驱动力,终身学习践行者,终身学习受益者。目前就职于国内某家知名互联网保险公司,担任DEVOPS工程师,对微服务领域和持续集成领域研究较深,精通微服务框架SpringCloud Spring Cloud Config Server最常见是将配置文件放在本地或者远程Git仓库,放在本地是将将所有的配置文件统一写在Config Server工程目录下,如果需要修改配置,需要重启config server;放在Git仓库,是将配置统一放在Git仓库,可以利用Git仓库的版本控制。本文将介绍使用另外一种方式存放配置信息,即将配置存放在Mysql中。整个流程:Config Sever暴露Http API接口,Config Client 通过调用Config Sever的Http API接口来读取配置Config Server的配置信息,Config Server从数据中读取具体的应用的配置。流程图如下:案例实战在本案例中需要由2个工程,分为config-server和config-client,其中config-server工程需要连接Mysql数据库,读取配置;config-client则在启动的时候从config-server工程读取。本案例Spring Cloud版本为Greenwich.RELEASE,Spring Boot版本为2.1.0.RELEASE。工程描述config-server端口8769,从数据库中读取配置config-client端口8083,从config-server读取配置搭建config-server工程创建工程config-server,在工程的pom文件引入config-server的起步依赖,mysql的连接器,jdbc的起步依赖,代码如下:1. <dependency>2. <groupId>org.springframework.cloud</groupId>3. <artifactId>spring-cloud-config-server</artifactId>4. </dependency>5. <dependency>6. <groupId>mysql</groupId>7. <artifactId>mysql-connector-java</artifactId>8. </dependency>9. <dependency>10. <groupId>org.springframework.boot</groupId>11. <artifactId>spring-boot-starter-jdbc</artifactId>12. </dependency>在工程的配置文件application.yml下做以下的配置:1. spring:2. profiles:3. active: jdbc4. application:5. name: config-jdbc-server6. datasource:7. url: jdbc:mysql://127.0.0.1:3306/config-jdbc?useUnicode=true&characterEncoding=utf8&characterSetResults=utf8&serverTimezone=GMT%2B88. username: root9. password: 12345610. driver-class-name: com.mysql.jdbc.Driver11. cloud:12. config:13. label: master14. server:15. jdbc: true16. server:17. port: 876918. spring.cloud.config.server.jdbc.sql: SELECT key1, value1 fromconfig_properties whereAPPLICATION=? andPROFILE=? andLABEL=?其中,spring.profiles.active为spring读取的配置文件名,从数据库中读取,必须为jdbc。spring.datasource配置了数据库相关的信息,spring.cloud.config.label读取的配置的分支,这个需要在数据库中数据对应。spring.cloud.config.server.jdbc.sql为查询数据库的sql语句,该语句的字段必须与数据库的表字段一致。在程序的启动文件ConfigServerApplication加上@EnableConfigServer注解,开启ConfigServer的功能,代码如下:1. @SpringBootApplication2. @EnableConfigServer3. publicclassConfigServerApplication{4. 5. publicstaticvoidmain(String[] args) {6. SpringApplication.run(ConfigServerApplication.class, args);7. }8. }初始化数据库由于Config-server需要从数据库中读取,所以读者需要先安装MySQL数据库,安装成功后,创建config-jdbc数据库,数据库编码为utf-8,然后在config-jdbc数据库下,执行以下的数据库脚本:1. CREATE TABLE `config_properties`(2. `id`bigint(20) NOT NULL AUTO_INCREMENT,3. `key1`varchar(50) COLLATE utf8_bin NOT NULL,4. `value1`varchar(500) COLLATE utf8_bin DEFAULT NULL,5. `application`varchar(50) COLLATE utf8_bin NOT NULL,6. `profile`varchar(50) COLLATE utf8_bin NOT NULL,7. `label`varchar(50) COLLATE utf8_bin DEFAULT NULL,8. PRIMARY KEY (`id`)9. ) ENGINE=InnoDBAUTO_INCREMENT=3DEFAULT CHARSET=utf8 COLLATE=utf8_bin其中key1字段为配置的key,value1字段为配置的值,application字段对应于应用名,profile对应于环境,label对应于读取的分支,一般为master。**数据config-client 的2条数据,包括server.port和foo两个配置,具体数据库脚本如下:1. insert into`config_properties`(`id`, `key1`, `value1`, `application`, `profile`, `label`) values('1','server.port','8083','config-client','dev','master');2. insert into`config_properties`(`id`, `key1`, `value1`, `application`, `profile`, `label`) values('2','foo','bar-jdbc','config-client','dev','master');搭建config-client在config-client工程的pom文件,引入web和config的起步依赖,代码如下:1. <dependency>2. <groupId>org.springframework.boot</groupId>3. <artifactId>spring-boot-starter-web</artifactId>4. </dependency>5. 6. <dependency>7. <groupId>org.springframework.cloud</groupId>8. <artifactId>spring-cloud-starter-config</artifactId>9. </dependency>在程序的启动配置文件bootstrap.yml做程序的相关配置,一定要是bootstrap.yml,不可以是application.yml,bootstrap.yml的读取优先级更高,配置如下:1. spring:2. application:3. name: config-client4. cloud:5. config:6. uri: http://localhost:87697. fail-fast: true8. profiles:9. active: dev其中spring.cloud.config.uri配置的config-server的地址,spring.cloud.config.fail-fast配置的是读取配置失败后,执行快速失败。spring.profiles.active配置的是spring读取配置文件的环境。在程序的启动文件ConfigClientApplication,写一个RestAPI,读取配置文件的foo配置,返回给浏览器,代码如下:1. @SpringBootApplication2. @RestController3. publicclassConfigClientApplication{4. 5. publicstaticvoidmain(String[] args) {6. SpringApplication.run(ConfigClientApplication.class, args);7. }8. 9. @Value("${foo}")10. Stringfoo;11. @RequestMapping(value = "/foo")12. publicStringhi(){13. returnfoo;14. }15. }依次启动2个工程,其中config-client的启动端口为8083,这个是在数据库中的,可见config-client从config-server中读取了配置。在浏览器**问http://localhost:8083/foo,浏览器显示bar-jdbc,这个是在数据库中的,可见config-client从config-server中读取了配置。参考资料https://cloud.spring.io/spring-cloud-config/single/spring-cloud-config.html#jdbcbackend源码下载https://github.com/forezp/SpringCloudLearning/tree/master/chapter10-5-jdbc 原创作者:方志朋方志朋简介:SpringCloud中国社区联合创始人,博客访问量突破一千万,爱好开源,热爱分享,活跃于各大社区,保持着非常强的学习驱动力,终身学习践行者,终身学习受益者。目前就职于国内某家知名互联网保险公司,担任DEVOPS工程师,对微服务领域和持续集成领域研究较深,精通微服务框架SpringCloud
-
什么是Nacos?Nacos 致力于帮助您发现、配置和管理微服务。Nacos 提供了一组简单易用的特性集,帮助您快速实现动态服务发现、服务配置、服务元数据及流量管理。是Spring Cloud A 中的服务注册发现组件,类似于Consul、Eureka,同时它又提供了分布式配置中心的功能,这点和Consul的config类似,支持热加载。Nacos 的关键特性包括:§ 服务发现和服务健康监测§ 动态配置服务,带管理界面,支持丰富的配置维度。§ 动态DNS 服务§ 服务及其元数据管理Nacos下载Nacos依赖于Java环境,所以必须安装Java环境。然后从官网下载Nacos的解压包,安装稳定版的,下载地址:https://github.com/alibaba/nacos/releases本次案例下载的版本为1.0.0 ,下载完成后,解压,在解压后的文件的/bin目录下,windows系统点击startup.cmd就可以启动nacos。linux或mac执行以下命令启动nacos。1. sh startup.sh -m standalone启动时会在控制台,打印相关的日志。nacos的启动端口为8848,在启动时要保证端口不被占用。珠穆拉马峰的高度是8844,nacos的端口是8848,有点巧合。启动成功,在浏览器**问:http://localhost:8848/nacos,会跳转到登陆界面,默认的登陆用户名为nacos,密码也为nacos。登陆成功后,展示的界面如下:从界面可知,此时没有服务注册到Nacos上。使用Nacos服务注册和发现服务注册和发现是微服务治理的根基,服务注册和发现组件是整个微服务系统的灵魂,选择合适的服务注册和发现组件至关重要,目前主流的服务注册和发现组件有Consul、Eureka、Etcd等。随着Eureka的闭源,Spring cloud netflix-oss组件大规模的进入到了维护期,不再提供新功能,spring cloud alibaba受到开源社区的大力拥护。服务注册在本案例中,使用2个服务注册到Nacos上,分别为nacos-provider和nacos-consumer。构建服务提供者nacos-provider新建一个Spring Boot项目,Spring boot版本为2.1.4.RELEASE,Spring Cloud 版本为Greenwich.RELEASE,在pom文件引入nacos的Spring Cloud起步依赖,代码如下:1. <dependency>2. <groupId>org.springframework.cloud</groupId>3. <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>4. <version>0.9.0.RELEASE</version>5. </dependency>在工程的配置文件application.yml做相关的配置,配置如下:1. server:2. port:87623. spring:4. application:5. name:nacos-provider6. cloud:7. nacos:8. discovery:9. server-addr:127.0.0.1:8848在上述的配置的中,程序的启动端口为8762,应用名为nacos-provider,向nacos server注册的地址为127.0.0.1:8848。然后在Spring Boot的启动文件NacosProviderApplication加上@EnableDiscoveryClient注解,代码如下:1. @SpringBootApplication2. @EnableDiscoveryClient3. publicclassNacosProviderApplication{4. 5. publicstaticvoidmain(String[]args){6. SpringApplication.run(NacosProviderApplication.class,args);7. }8. 9. }构建服务消费者nacos-consuer和nacos-provider一样,构建服务消费者nacos-consumer,nacos-cosumer的启动端口8763。构建过程同nacos-provider,这里省略。验证服务注册个发现分别启动2个工程,待工程启动成功之后,在访问localhost:8848,可以发现nacos-provider和nacos-consumer,均已经向nacos-server注册,如下图所示:服务调用nacos作为服务注册和发现组件时,在进行服务消费,可以选择RestTemplate和Feign等方式。这和使用Eureka和Consul作为服务注册和发现的组件是一样的,没有什么区别。这是因为spring-cloud-starter-alibaba-nacos-discovery依赖实现了Spring Cloud服务注册和发现的相关接口,可以和其他服务注册发现组件无缝切换。提供服务在nacos-provider工程,写一个Controller提供API服务,代码如下:1. @RestController2. publicclassProviderController{3. 4. Loggerlogger=LoggerFactory.getLogger(ProviderController.class);5. 6. @GetMapping("/hi")7. publicStringhi(@RequestParam(value ="name",defaultValue ="forezp",required =false)Stringname){8. 9. return"hi "+name;10. }11. }消费服务在这里使用2种方式消费服务,一种是RestTemplate,一种是Feign。使用RestTemplate消费服务RestTemplate可以使用Ribbon作为负载均衡组件,在nacos-consumer工程中引入ribbon的依赖:1. <dependency>2. <groupId>org.springframework.cloud</groupId>3. <artifactId>spring-cloud-starter-ribbon</artifactId>4. </dependency>在NacosConsumerApplication启动文件注入RestTemplate的Bean,代码如下:1. @LoadBalanced2. @Bean3. publicRestTemplaterestTemplate(){4. returnnewRestTemplate();5. }加上@LoadBalanced注解即可在RestTemplate上开启LoadBalanced负载均衡的功能。写一个消费服务的ConsumerController,代码如下:1. @RestController2. publicclassConsumerController{3. 4. @Autowired5. RestTemplaterestTemplate;6. 7. @GetMapping("/hi-resttemplate")8. publicStringhiResttemplate(){9. returnrestTemplate.getForObject("http://nacos-provider/hi?name=resttemplate",String.class);10. 11. }重启工程,在浏览器**问http://localhost:8763/hi-resttemplate,可以在浏览器上展示正确的响应,这时nacos-consumer调用nacos-provider服务成功。是FeignClient调用服务在nacos-consumer的pom文件引入以下的依赖:1. <dependency>2. <groupId>org.springframework.cloud</groupId>3. <artifactId>spring-cloud-starter-openfeign</artifactId>4. </dependency>在NacosConsumerApplication启动文件上加上@EnableFeignClients注解开启FeignClient的功能。1. @EnableFeignClients2. publicclassNacosConsumerApplication{3. 4. publicstaticvoidmain(String[]args){5. SpringApplication.run(NacosConsumerApplication.class,args);6. }写一个FeignClient,调用nacos-provider的服务,代码如下:1. @FeignClient("nacos-provider")2. publicinterfaceProviderClient{3. 4. @GetMapping("/hi")5. Stringhi(@RequestParam(value ="name",defaultValue ="forezp",required =false)Stringname);6. }写一个消费API,该API使用ProviderClient来调用nacos-provider的API服务,代码如下:1. @RestController2. publicclassConsumerController{3. 4. 5. @Autowired6. ProviderClientproviderClient;7. 8. @GetMapping("/hi-feign")9. publicStringhiFeign(){10. returnproviderClient.hi("feign");11. }12. }重启工程,在浏览器**问http://localhost:8763/hi-feign,可以在浏览器上展示正确的响应,这时nacos-consumer调用nacos-provider服务成功。总结本文比较详细的介绍了如何使用Nacos作为服务注册中心,并使用案例介绍了如何在使用nacos作为服务注册中心时消费服务。下一篇教程将介绍如何使用nacos作为分布式配置中心。源码下载https://github.com/forezp/SpringCloudLearning/tree/master/springcloud-alibaba/nacos-discovery参考资料https://nacos.io/zh-cn/docs/what-is-nacos.html 原创作者:方志朋方志朋简介:SpringCloud中国社区联合创始人,博客访问量突破一千万,爱好开源,热爱分享,活跃于各大社区,保持着非常强的学习驱动力,终身学习践行者,终身学习受益者。目前就职于国内某家知名互联网保险公司,担任DEVOPS工程师,对微服务领域和持续集成领域研究较深,精通微服务框架Spring Cloud
上滑加载中
推荐直播
-
算子工具性能优化新特性演示——MatMulLeakyRelu性能调优实操
2025/01/10 周五 15:30-17:30
MindStudio布道师
算子工具性能优化新特性演示——MatMulLeakyRelu性能调优实操
即将直播 -
用代码全方位驱动 OBS 存储
2025/01/14 周二 16:30-18:00
阿肯 华为云生态技术讲师
如何用代码驱动OBS?常用的数据管理,对象清理,多版本对象访问等应该如何编码?本期课程一一演示解答。
即将直播 -
GaussDB数据库开发
2025/01/15 周三 16:00-17:30
Steven 华为云学堂技术讲师
本期直播将带你了解GaussDB数据库开发相关知识,并通过实验指导大家利用java基于JDBC的方式来完成GaussD数据库基础操作。
去报名
热门标签