这篇文章主要讲述服务追踪组件zipkin,Spring Cloud Sleuth集成了zipkin组件。

一、简介

Spring Cloud Sleuth 主要功能就是在分布式系统中提供追踪解决方案,并且兼容支持了 zipkin,你只需要在pom文件中引入相应的依赖即可。

二、服务追踪分析

微服务架构上通过业务来划分服务的,通过REST调用,对外暴露的一个接口,可能需要很多个服务协同才能完成这个接口功能,如果链路上任何一个服务出现问题或者网络超时,都会形成导致接口调用失败。随着业务的不断扩张,服务之间互相调用会越来越复杂。

随着服务的越来越多,对调用链的分析会越来越复杂。它们之间的调用关系也许如下:

三、术语

  • Span:基本工作单元,例如,在一个新建的span中发送一个RPC等同于发送一个回应请求给RPC,span通过一个64位ID唯一标识,trace以另一个64位ID表示,span还有其他数据信息,比如摘要、时间戳事件、关键值注释(tags)、span的ID、以及进度ID(通常是IP地址) 
    span在不断的启动和停止,同时记录了时间信息,当你创建了一个span,你必须在未来的某个时刻停止它。
  • Trace:一系列spans组成的一个树状结构,例如,如果你正在跑一个分布式大数据工程,你可能需要创建一个trace。
  • Annotation:用来及时记录一个事件的存在,一些核心annotations用来定义一个请求的开始和结束 
    • cs - Client Sent -客户端发起一个请求,这个annotion描述了这个span的开始
    • sr - Server Received -服务端获得请求并准备开始处理它,如果将其sr减去cs时间戳便可得到网络延迟
    • ss - Server Sent -注解表明请求处理的完成(当请求返回客户端),如果ss减去sr时间戳便可得到服务端需要的处理请求时间
    • cr - Client Received -表明span的结束,客户端成功接收到服务端的回复,如果cr减去cs时间戳便可得到客户端从服务端获取回复的所有所需时间 
      将Span和Trace在一个系统中使用Zipkin注解的过程图形化:

将Span和Trace在一个系统中使用Zipkin注解的过程图形化:

四、构建工程

基本知识讲解完毕,下面我们来实战,本文的案例主要有三个工程组成:

1.server-zipkin

它的主要作用使用ZipkinServer 的功能,收集调用数据,并展示;

2.chetxt-service

对外暴露接口

3.gps-service

对外暴露接口

上面两个service可以相互调用,并且只有调用了,server-zipkin才会收集数据的,这就是为什么叫服务追踪了。

新建server-zipkin

pom.xml

<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-zipkin</artifactId>
</dependency>
<dependency>
<groupId>io.zipkin.java</groupId>
<artifactId>zipkin-server</artifactId>
</dependency> <dependency>
<groupId>io.zipkin.java</groupId>
<artifactId>zipkin-autoconfigure-ui</artifactId>
</dependency>

springboot启动类

@SpringBootApplication(exclude = {DataSourceAutoConfiguration.class}) // 不加红字会出现默认database的错误
@EnableZipkinServer
public class BootApplication {
public static void main(String[] args) {
SpringApplication.run(BootApplication.class, args);
}
}

配置文件

server.port=9411

新建chetxt-service

pom.xml

<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-zipkin</artifactId>
</dependency>

配置文件

server.port=8008

spring.zipkin.base-url=http://localhost:9411
spring.application.name=chetxt-service

springboot启动类

@SpringBootApplication
@ComponentScan(basePackages="com.chetxt")//扫描组件
@ServletComponentScan(basePackages="com.chetxt")//扫描拦截器,过滤器
public class BootApplication {
public static void main(String[] args) {
SpringApplication.run(BootApplication.class, args);
} @Bean
public RestTemplate restTemplate() {
HttpComponentsClientHttpRequestFactory httpRequestFactory = new HttpComponentsClientHttpRequestFactory();
httpRequestFactory.setConnectionRequestTimeout(10000);
httpRequestFactory.setConnectTimeout(10000);
httpRequestFactory.setReadTimeout(10000);
return new RestTemplate();//httpRequestFactory);
} @Bean // 采样器
public AlwaysSampler defaultSampler(){
return new AlwaysSampler();
}
}

暴露接口

@RestController
@RequestMapping("/v1/vehicleorder")
public class OrderinfoController {
@RequestMapping(value = "/addgps", method = RequestMethod.GET)
// @Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.DEFAULT, readOnly = false)
public String addgps(String Appcode) {
// 验证笔录项目
if(Appcode == null || Appcode.equals("")){
TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
return new ZhongruiResult(false, true, "缺少appcode", null, null).toJsonString();
} JSONObject param = new JSONObject();
try {
param.put("appcode", Appcode);
} catch (JSONException e) {
e.printStackTrace();
}

     // resttemplate调用gps-service暴露的接口
HttpHeaders headers = new HttpHeaders();
MediaType type = MediaType.parseMediaType("application/json; charset=UTF-8");
headers.setContentType(type);
headers.add("Accept", MediaType.APPLICATION_JSON_UTF8_VALUE.toString());//application/json
HttpEntity<String> formEntity = new HttpEntity<String>(param.toString(), headers);
String rtnStr=this.restTemplate.postForObject("http://localhost:8009/gps/order/add", formEntity,String.class); return rtnStr;
}   
@RequestMapping(value = "/xxx", method = RequestMethod.GET)
// @Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.DEFAULT, readOnly = false)
public String xxx(String Appcode) {   }
  
}

创建gps-service

pom文件和启动类和上面一样

暴露接口

@RestController
@RequestMapping("/gps/order")
public class OrderinfoController {
public static Logger logger = LoggerFactory.getLogger(OrderinfoController.class); @Autowired
private RestTemplate restTemplate; @Value("${server.port}")
private int port; @RequestMapping(value = "/add", method = RequestMethod.POST, consumes = "application/json", produces = "*/*")
// @Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.DEFAULT, readOnly = false)
public String add(@RequestBody RequestAdd param) { // TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
return new ObjectResult(true, "", JsonUtil.object2JsonString(param), null).toJsonString();
} }

配置文件

server.port=8009

spring.zipkin.base-url=http://localhost:9411
spring.application.name=gps-service

一切准备就绪,下面启动三个微服务。

五、演示

浏览器里输入下面的地址,模拟服务之间调用接口

http://localhost:8008/v1/vehicleorder/addgps?Appcode=112233445566

返回结果,表示调用成功

打开zipkin的追踪网址

http://localhost:9411

点击2 spans那一行

点击链接行,弹出具体信息

上面是成功了的,下面改下代码,让gps-service里的add接口出现访问超时

@Bean
public RestTemplate restTemplate() {
HttpComponentsClientHttpRequestFactory httpRequestFactory = new HttpComponentsClientHttpRequestFactory();
httpRequestFactory.setConnectionRequestTimeout(10000);
httpRequestFactory.setConnectTimeout(10000);
httpRequestFactory.setReadTimeout(10000);
return new RestTemplate(httpRequestFactory);// 设置resttemplate访问10s超时
}
@RequestMapping(value = "/add", method = RequestMethod.POST, consumes = "application/json", produces = "*/*")
// @Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.DEFAULT, readOnly = false)
public String add(@RequestBody RequestAdd param) {
// 睡眠30s
for (int i = 0; i < 30; i++) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
} // TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
return new ObjectResult(true, "", JsonUtil.object2JsonString(param), null).toJsonString();
}

再次请求地址,报超时错误,这是我们预期的

再去zipkin里看看

点击红色链接进来

再点击查看具体原因

好了,是不是非常棒,再也不用怕服务一多乱套了。

源码在微云。

原文地址:http://blog.csdn.net/forezp/article/details/70162074

参考文章:

spring-cloud-sleuth+zipkin追踪服务实现(一)

spring-cloud-sleuth+zipkin追踪服务实现(二)

spring-cloud-sleuth+zipkin追踪服务实现(三)

Spring Cloud(十三):Spring Cloud Sleuth服务链路追踪(zipkin)(转)的更多相关文章

  1. Spring Cloud Sleuth服务链路追踪(zipkin)(转)

    这篇文章主要讲述服务追踪组件zipkin,Spring Cloud Sleuth集成了zipkin组件. 一.简介 Spring Cloud Sleuth 主要功能就是在分布式系统中提供追踪解决方案, ...

  2. 服务链路追踪(Spring Cloud Sleuth)

    sleuth:英 [slu:θ] 美 [sluθ] n.足迹,警犬,侦探vi.做侦探 微服务架构是一个分布式架构,它按业务划分服务单元,一个分布式系统往往有很多个服务单元.由于服务单元数量众多,业务的 ...

  3. spring cloud 入门系列八:使用spring cloud sleuth整合zipkin进行服务链路追踪

    好久没有写博客了,主要是最近有些忙,今天忙里偷闲来一篇. =======我是华丽的分割线========== 微服务架构是一种分布式架构,微服务系统按照业务划分服务单元,一个微服务往往会有很多个服务单 ...

  4. Spring Cloud Sleuth+ZipKin+ELK服务链路追踪(七)

    序言 sleuth是spring cloud的分布式跟踪工具,主要记录链路调用数据,本身只支持内存存储,在业务量大的场景下,为拉提升系统性能也可通过http传输数据,也可换做rabbit或者kafka ...

  5. SpringCloud学习之Sleuth服务链路跟踪(十二)

    一.为什么需要Spring Cloud Sleuth 微服务架构是一个分布式架构,它按业务划分服务单元,一个分布式系统往往有很多个服务单元.由于服务单元数量众多,业务的复杂性,如果出现了错误和异常,很 ...

  6. Gokit微服务-服务链路追踪

    https://mp.weixin.qq.com/s/gjKOy4SDpsjUXDC3Q1YdFw Gokit微服务-服务链路追踪 原创: 兮一昂吧 兮一昂吧 2月28日

  7. go微服务框架kratos学习笔记九(kratos 全链路追踪 zipkin)

    目录 go微服务框架kratos学习笔记九(kratos 全链路追踪 zipkin) zipkin使用demo 数据持久化 go微服务框架kratos学习笔记九(kratos 全链路追踪 zipkin ...

  8. SpringCloud(7)服务链路追踪Spring Cloud Sleuth

    1.简介 Spring Cloud Sleuth 主要功能就是在分布式系统中提供追踪解决方案,并且兼容支持了 zipkin,你只需要在pom文件中引入相应的依赖即可.本文主要讲述服务追踪组件zipki ...

  9. spring boot 2.0.3+spring cloud (Finchley)7、服务链路追踪Spring Cloud Sleuth

    参考:Spring Cloud(十二):分布式链路跟踪 Sleuth 与 Zipkin[Finchley 版] Spring Cloud Sleuth 是Spring Cloud的一个组件,主要功能是 ...

随机推荐

  1. jquery 获取被点击元素的id属性值

    有时候可能需要获取被点击元素的一些信息,此处就以id属性为例子,进行演示一下. $(document).click(function (e){ var v_id=e.target.id; consol ...

  2. Java EE 学习(9):IDEA + maven + spring 搭建 web(5)- 博客文章管理

    转载:Gaussic(一个致力于AI研究却不得不兼顾项目的研究生) . 注:在阅读本文前,请先阅读: Java EE 学习(5):IDEA + maven + spring 搭建 web(1) Jav ...

  3. 【09】react 之 表单组件

    不太清楚有多少初学React的同学和博主当时一样,在看完React的生命周期.数据流之后觉得已经上手了,甩开文档啪啪啪的开始敲了起来.结果...居然被一个input标签给教做人了. 故事是这样的:首先 ...

  4. java--Eclipse for mac 代码提示(代码助手,代码联想)快捷键修改

    Eclipse for mac 代码提示(代码助手,代码联想)快捷键修改 一.每次输入都自动提示 点击Eclipse,使其成为第一响应者,preferences->Java->Editor ...

  5. 洛谷 [P1337] 平衡点

    模拟退火练手 一道模拟退火的好题 结果一定势能最小 与模拟退火思路高度一致 #include <iostream> #include <cstdio> #include < ...

  6. UVa 11234 The Largest Clique

    找最长的连接的点的数量.用tarjan缩点,思考可知每一个强连通分量里的点要么都选,要么都不选(走别的路),可以动规解决. #include<iostream> #include<c ...

  7. 【HDOJ5976】Detachment(贪心)

    题意:给定n,要求构造若干个各不相同且和为n的正整数使得它们的乘积最大 T<=1e6,1<=n<=1e9 思路:From https://blog.csdn.net/qq_34374 ...

  8. C#可选参数与具名参数

    可选参数 static void test1() { func1("A"); func1(); Console.ReadKey(); } ) { Console.WriteLine ...

  9. js 字符串长度截取

    <script> function cutstr(str, len) { var temp, icount = 0, patrn = /[^\x00-\xff]/, strre = &qu ...

  10. The C programming language [book]

    1 A12.3 Macro Definition and Expansion A control line of the form #define identifier token-sequence ...