skywalking09 - 异步线程链路续接(上)

简介: skywalking09 - 异步线程链路续接(上)

skywalking09 - 异步线程链路续接(上)

不知道你是不是也出现过,明明打了@Trace,但是死活从http请求的链路进来看不到,反而,它还独自成一条链路,这样一来,根本连不成一个链路来追踪问题了。

断掉的链路

异步代码

此处,用了线程池去异步执行一个实现Runnable接口的类。

@Trace
    public void trace() throws InterruptedException {
        Thread.sleep(10);
        doNothing();
        executor.submit(new MyRunnable());
    }

以及该接口具体实现

public class MyRunnable implements Runnable{
    @Override
    public void run() {
        System.out.println(TraceContext.traceId());
        doNothing();
    }
    @Trace
    private void doNothing(){
        return;
    }
}

链路断开示意图

理想中,我们自然也是希望异步线程中@Trace加注的方法也进入对应的链路,但是很遗憾,链路断成两条了:

MyRunnable中的doNothing并没有在/trace/local这个http请求中打印出来。

连上的链路

链路连上示意图

我们会发现,链路仍然有两条存在,但是,/trace/local这个http请求中打印出来多了两个Span,一个是异步线程,一个是添加了@Trace的自定义的方法。而且,这两条链路的链路流水号是同一个。

实现方式

  • @TraceCrossThread
  • @Async
  • apm-jdk-threading-plugin

三种方式都可以实现,稍微做点比较。

@TraceCrossThread

@TraceCrossThread为skywalking提供的工具包中的注解,你可以认为其有一定的侵入性。使用方式则是在那个线程类上加上该注解,在类加载时,其构造方法会被代理agent做一次增强,具体的源码下一节说。

@TraceCrossThread
public class MyRunnable implements Runnable{
    @Override
    public void run() {
        System.out.println(TraceContext.traceId());
        doNothing();
    }
}

@Async

@Async为spring-context包下的注解,对于java服务端来说,这个注解基本可以算无侵入了,毕竟是spring体系。不过使用方式和平时需要实现一个Runnable接口或继承Thread类差别挺大,个人觉得方便,但是在古董项目中、亦或是说服那些一年学习经验十年使用经验的老古董来说,推广起来还是比较头疼的。当然,如果是一个巨石的老项目,不推荐这样来改造了。

@Service
public class TraceService implements InitializingBean {
    @Trace
    @Async
    public void traceAsync() throws InterruptedException {
        Thread.sleep(10);
        doNothing();
    }
    @Trace
    public void trace() throws InterruptedException {
        Thread.sleep(10);
        doNothing();
        executor.submit(new MyRunnable());
    }
}

关于这个注解的详细使用可以见:JAVA多线程以及Spring异步注解@Async。效果图如下,其会多一个SpringAsync的链路,当然其链路流水号也是相同的:

apm-jdk-threading-plugin

apm-jdk-threading-plugin这是官方提供的一个插件,是真正的无侵入了,使用方式也很简单,这个插件位于${skywalking_dir}/agent/bootstrap-plugins目录下,我们需要做的就是将其复制到${skywalking_dir}/agent/plugins目录下即可。

另外,还需要对代理的配置进行修改${skywalking_dir}/agent/config/agent.config,告诉代理对那些包下的线程池进行增强,在其底部添加这样的配置:

jdkthreading.threading_class_prefixes=com.aaa.bbb,com.bbb.ccc

当有多个包的时候,可以用英文的逗号进行分隔。

总结

多线程可能会导致链路断开,我们的链路续接不上,很多问题就没有办法查,要根据合适的方式,将链路接上。

目录
相关文章
|
1月前
|
编解码 数据安全/隐私保护 计算机视觉
Opencv学习笔记(十):同步和异步(多线程)操作打开海康摄像头
如何使用OpenCV进行同步和异步操作来打开海康摄像头,并提供了相关的代码示例。
78 1
Opencv学习笔记(十):同步和异步(多线程)操作打开海康摄像头
|
1月前
|
安全 调度 C#
STA模型、同步上下文和多线程、异步调度
【10月更文挑战第19天】本文介绍了 STA 模型、同步上下文和多线程、异步调度的概念及其优缺点。STA 模型适用于单线程环境,确保资源访问的顺序性;同步上下文和多线程提高了程序的并发性和响应性,但增加了复杂性;异步调度提升了程序的响应性和资源利用率,但也带来了编程复杂性和错误处理的挑战。选择合适的模型需根据具体应用场景和需求进行权衡。
|
1月前
|
网络协议 安全 Java
难懂,误点!将多线程技术应用于Python的异步事件循环
难懂,误点!将多线程技术应用于Python的异步事件循环
61 0
|
2月前
|
设计模式 缓存 Java
谷粒商城笔记+踩坑(14)——异步和线程池
初始化线程的4种方式、线程池详解、异步编排 CompletableFuture
谷粒商城笔记+踩坑(14)——异步和线程池
|
3月前
|
缓存 Java
异步&线程池 线程池的七大参数 初始化线程的4种方式 【上篇】
这篇文章详细介绍了Java中线程的四种初始化方式,包括继承Thread类、实现Runnable接口、实现Callable接口与FutureTask结合使用,以及使用线程池。同时,还深入探讨了线程池的七大参数及其作用,解释了线程池的运行流程,并列举了四种常见的线程池类型。最后,阐述了在开发中使用线程池的原因,如降低资源消耗、提高响应速度和增强线程的可管理性。
异步&线程池 线程池的七大参数 初始化线程的4种方式 【上篇】
|
3月前
|
Java 数据库
异步&线程池 CompletableFuture 异步编排 实战应用 【终结篇】
这篇文章通过一个电商商品详情页的实战案例,展示了如何使用`CompletableFuture`进行异步编排,以解决在不同数据库表中查询商品信息的问题,并提供了详细的代码实现和遇到问题(如图片未显示)的解决方案。
异步&线程池 CompletableFuture 异步编排 实战应用 【终结篇】
|
3月前
|
Java
异步&线程池 CompletableFuture 异步编排 【下篇】
这篇文章深入探讨了Java中的`CompletableFuture`类,解释了如何创建异步操作、使用计算完成时的回调方法、异常处理、串行化方法、任务组合以及多任务组合的使用方式,并通过代码示例展示了各种场景下的应用。
异步&线程池 CompletableFuture 异步编排 【下篇】
|
3月前
|
数据采集 Python
多线程和异步
【8月更文挑战第12天】
39 3
|
4月前
|
Java Spring 容器
Spring boot 自定义ThreadPoolTaskExecutor 线程池并进行异步操作
Spring boot 自定义ThreadPoolTaskExecutor 线程池并进行异步操作
211 3
|
3月前
|
Dart API C语言
Dart ffi 使用问题之想在C/C++中创建异步线程来调用Dart方法,如何操作
Dart ffi 使用问题之想在C/C++中创建异步线程来调用Dart方法,如何操作