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天前
|
缓存 Java
异步&线程池 线程池的七大参数 初始化线程的4种方式 【上篇】
这篇文章详细介绍了Java中线程的四种初始化方式,包括继承Thread类、实现Runnable接口、实现Callable接口与FutureTask结合使用,以及使用线程池。同时,还深入探讨了线程池的七大参数及其作用,解释了线程池的运行流程,并列举了四种常见的线程池类型。最后,阐述了在开发中使用线程池的原因,如降低资源消耗、提高响应速度和增强线程的可管理性。
异步&线程池 线程池的七大参数 初始化线程的4种方式 【上篇】
|
1天前
|
Java 数据库
异步&线程池 CompletableFuture 异步编排 实战应用 【终结篇】
这篇文章通过一个电商商品详情页的实战案例,展示了如何使用`CompletableFuture`进行异步编排,以解决在不同数据库表中查询商品信息的问题,并提供了详细的代码实现和遇到问题(如图片未显示)的解决方案。
异步&线程池 CompletableFuture 异步编排 实战应用 【终结篇】
|
1天前
|
Java
异步&线程池 CompletableFuture 异步编排 【下篇】
这篇文章深入探讨了Java中的`CompletableFuture`类,解释了如何创建异步操作、使用计算完成时的回调方法、异常处理、串行化方法、任务组合以及多任务组合的使用方式,并通过代码示例展示了各种场景下的应用。
异步&线程池 CompletableFuture 异步编排 【下篇】
|
8天前
|
数据采集 Python
多线程和异步
【8月更文挑战第12天】
19 3
|
23天前
|
Java Spring 容器
Spring boot 自定义ThreadPoolTaskExecutor 线程池并进行异步操作
Spring boot 自定义ThreadPoolTaskExecutor 线程池并进行异步操作
46 3
|
4天前
|
Dart API C语言
Dart ffi 使用问题之想在C/C++中创建异步线程来调用Dart方法,如何操作
Dart ffi 使用问题之想在C/C++中创建异步线程来调用Dart方法,如何操作
|
1月前
|
Java
不自定义异步方法的线程池默认使用SimpleAsyncTaskExecutor
不自定义异步方法的线程池默认使用SimpleAsyncTaskExecutor
|
1月前
|
安全 Java 数据库连接
Spring Boot 优雅关机时异步线程安全优化
Spring Boot 优雅关机时异步线程安全优化
37 1
|
1月前
|
安全 NoSQL Java
网络安全-----Redis12的Java客户端----客户端对比12,Jedis介绍,使用简单安全性不足,lettuce(官方默认)是基于Netty,支持同步,异步和响应式,并且线程是安全的,支持R
网络安全-----Redis12的Java客户端----客户端对比12,Jedis介绍,使用简单安全性不足,lettuce(官方默认)是基于Netty,支持同步,异步和响应式,并且线程是安全的,支持R
|
2月前
|
监控 安全 Java
Spring Boot优雅Shutdown时异步线程安全优化
Spring Boot优雅Shutdown时异步线程安全优化