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

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

总结

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

目录
相关文章
|
4月前
|
Python
【Python30天速成计划】10.异步以及多进程和多线程
【Python30天速成计划】10.异步以及多进程和多线程
|
2月前
|
Python
Python学习之路 02 之分支结构
Python学习之路 02 之分支结构
457 0
Python学习之路 02 之分支结构
|
2月前
|
Java Python 开发者
Python 学习之路 01基础入门---【Python安装,Python程序基本组成】
线程池详解与异步任务编排使用案例-xian-cheng-chi-xiang-jie-yu-yi-bu-ren-wu-bian-pai-shi-yong-an-li
473 2
Python 学习之路 01基础入门---【Python安装,Python程序基本组成】
|
1月前
|
JavaScript 前端开发
JS 单线程还是多线程,如何显示异步操作
JS 单线程还是多线程,如何显示异步操作
22 2
|
2月前
|
JavaScript Java API
spring boot使用异步多线程
一文讲清楚spring boot如何结合异步多线程实现文件的导出这类耗时间的操作优化以及常用的场景,了解异步思想
39 0
spring boot使用异步多线程
|
2月前
|
Java
多线程------Future异步任务
多线程------Future异步任务
|
4月前
|
iOS开发
多线程和异步编程:解释 iOS 中的同步和异步任务的概念。
多线程和异步编程:解释 iOS 中的同步和异步任务的概念。
40 1
|
5月前
|
Java
Java异步、线程池解决方案
Java异步、线程池解决方案
47 0
|
5月前
|
XML Java 调度
Android App网络通信中通过runOnUiThread快速操纵界面以及利用线程池Executor调度异步任务实战(附源码 简单易懂)
Android App网络通信中通过runOnUiThread快速操纵界面以及利用线程池Executor调度异步任务实战(附源码 简单易懂)
31 0
|
5月前
|
Java Spring
SpringBoot - @Async异步任务与线程池
SpringBoot - @Async异步任务与线程池
70 1