新闻中心
OpenTelemetry J*a:利用上下文传播构建分布式 Span 关系

本文详细阐述了在 opentelemetry j*a 中如何基于 span id 实现分布式追踪的上下文传播。重点介绍了 opentelemetry 不直接通过 span id 获取 span 对象的设计理念,而是通过注入(inject)和提取(extract)操作,将追踪上下文(包括父 span id 和 trace id)在服务间传递,从而正确建立父子 span 关系,确保分布式系统中追踪链的完整性。
在分布式系统中,服务间的调用链路追踪是实现可观测性的关键。OpenTelemetry 提供了一套强大的 API 和 SDK 来实现这一目标。开发者在使用 OpenTelemetry J*a 进行追踪时,有时会遇到一个常见问题:如何在一个服务中,仅凭父 Span 的 ID 来“获取”或“引用”该父 Span,以便为当前操作设置正确的父子关系?
OpenTelemetry 的设计哲学是,Span 对象是特定执行上下文的瞬时表示。它不提供一个全局注册表或方法,允许用户通过 Span ID 直接检索一个 Span 对象。这是因为 Span 的生命周期通常与它所代表的操作绑定,并且在操作完成后即结束。当涉及到跨进程或跨服务边界的追踪时,OpenTelemetry 依赖于上下文传播(Context Propagation)机制来传递追踪信息,而非直接传递 Span 对象本身。
核心概念:上下文传播 (Context Propagation)
上下文传播是 OpenTelemetry 中处理分布式追踪父子关系的核心机制。它允许追踪上下文(SpanContext,其中包含 Trace ID 和 Span ID)在不同的服务或进程之间传递。这个过程通常分为两个主要步骤:
- 注入 (Inject):在调用方服务中,将当前 Span 的 SpanContext 序列化并注入到一个传输载体(如 HTTP 请求头、消息队列的元数据)中。
- 提取 (Extract):在被调用方服务中,从接收到的传输载体中反序列化出 SpanContext,并基于此上下文创建新的子 Span。
通过这种方式,即使父 Span 对象在调用方已经结束,其关键的追踪信息(Trace ID 和 Span ID)也能被传递到下游服务,从而在下游服务中正确地建立起与上游服务的父子关系,形成完整的分布式追踪链。
立即学习“J*a免费学习笔记(深入)”;
OpenTelemetry J*a 中的上下文传播实现
OpenTelemetry J*a SDK 提供了 TextMapPropagator 接口来处理上下文的注入和提取。W3C Trace Context 是推荐的跨服务追踪上下文传播标准,OpenTelemetry 默认支持并推荐使用 W3CTraceContextPropagator。
1. 注入(Injection):在调用方服务中传递上下文
在调用方服务中,您需要获取当前活跃的 Span 的上下文,并将其注入到即将发送的请求中。
Lateral App
整理归类论文
85
查看详情
import io.opentelemetry.api.OpenTelemetry;
import io.opentelemetry.api.trace.Span;
import io.opentelemetry.api.trace.Tracer;
import io.opentelemetry.context.Context;
import io.opentelemetry.context.Scope;
import io.opentelemetry.api.trace.propagation.W3CTraceContextPropagator;
import io.opentelemetry.context.propagation.TextMapSetter;
import j*a.util.HashMap;
import j*a.util.Map;
public class CallerService {
private final Tracer tracer;
private final OpenTelemetry openTelemetry;
public CallerService(OpenTelemetry openTelemetry) {
this.openTelemetry = openTelemetry;
this.tracer = openTelemetry.getTracer("my-caller-service");
}
/**
* 模拟一个远程调用,并将当前 Span 的上下文注入到传输载体中。
* @return 包含追踪上下文的 Map,模拟 HTTP Headers。
*/
public Map<String, String> makeRemoteCall() {
Span parentSpan = tracer.spanBuilder("parentSpanInCaller")
.startSpan();
Map<String, String> carrier = new HashMap<>(); // 模拟 HTTP Headers 或消息队列元数据
try (Scope scope = parentSpan.makeCurrent()) {
// 获取当前上下文,并使用 W3CTraceContextPropagator 将其注入到 carrier 中
W3CTraceContextPropagator.getInstance().inject(
Context.current(), // 获取当前活跃的 Span 所在的上下文
carrier, // 传输载体,例如一个 Map 来模拟 HTTP Headers
new TextMapSetter<Map<String, String>>() {
@Override
public void set(Map<String, String> carrier, String key, String value) {
carrier.put(key, value);
}
});
System.out.println("调用方 Span ID: " + parentSpan.getSpanContext().getSpanId());
System.out.println("注入的追踪上下文: " + carrier);
} finally {
parentSpan.end();
}
return carrier;
}
}在上述代码中:
- 我们创建了一个名为 parentSpanInCaller 的 Span,并将其设置为当前活跃 Span。
- W3CTraceContextPropagator.getInstance().inject() 方法负责从 Context.current() 中提取 SpanContext 信息,并将其写入到 carrier(这里是一个 Map
)中。TextMapSetter 接口定义了如何将键值对写入到载体中。 - carrier 随后会被发送到被调用方服务。
2. 提取(Extraction):在被调用方服务中重建上下文
在被调用方服务中,您需要从接收到的请求中提取追踪上下文,并用它来作为新创建 Span 的父级。
import io.opentelemetry.api.OpenTelemetry;
import io.opentelemetry.api.trace.Span;
import io.opentelemetry.api.trace.Tracer;
import io.opentelemetry.context.Context;
import io.opentelemetry.api.trace.propagation.W3CTraceContextPropagator;
import io.opentelemetry.context.propagation.TextMapGetter;
import j*a.util.Collections;
import j*a.util.Map;
public class CalleeService {
private final Tracer tracer;
priva
te final OpenTelemetry openTelemetry;
public CalleeService(OpenTelemetry openTelemetry) {
this.openTelemetry = openTelemetry;
this.tracer = openTelemetry.getTracer("my-callee-service");
}
/**
* 模拟处理远程调用,从传输载体中提取上下文,并创建子 Span。
* @param carrier 包含追踪上下文的 Map,模拟 HTTP Headers。
*/
public void processRemoteCall(Map<String, String> carrier) {
// 从 carrier 中提取追踪上下文
Context extractedContext = W3CTraceContextPropagator.getInstance().extract(
Context.current(), // 默认上下文,如果 carrier 中没有追踪信息,则使用此上下文
carrier, // 传输载体
new TextMapGetter<Map<String, String>>() {
@Override
public Iterable<String> keys(Map<String, String> carrier) {
return carrier.keySet();
}
@Override
public String get(Map<String, String> carrier, String key) {
return carrier.get(key);
}
});
// 使用提取到的上下文作为新 Span 的父级
Span childSpan = tracer.spanBuilder("childSpanInCallee")
.setParent(extractedContext) // 将提取到的上下文设置为父级
.startSpan();
try {
System.out.println("被调用方 Span ID: " + childSpan.getSpanContext().getSpanId());
System.out.println("被调用方父 Span ID: " + childSpan.getParentSpanContext().getSpanId());
// 模拟一些工作
Thread.sleep(100);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
} finally {
childSpan.end();
}
}
}在上述代码中:
- W3CTraceContextPropagator.getInstance().extract() 方法从 carrier 中读取追踪上下文信息,并返回一个 Context 对象。TextMapGetter 接口定义了如何从载体中读取键值对。
- 这个 extractedContext 对象包含了上游服务 Span 的 SpanContext。
- 通过 tracer.spanBuilder("childSpanInCallee").setParent(extractedContext).startSpan(),我们创建了一个新的 Span,并明确指定其父级为 extractedContext 中携带的 Span。这样就成功建立了跨服务的父子关系。
完整的端到端示例
为了演示上述过程,我们需要一个主程序来初始化 OpenTelemetry SDK,并协调调用方和被调用方服务。
import io.opentelemetry.api.OpenTelemetry;
import io.opentelemetry.sdk.OpenTelemetrySdk;
import io.opentelemetry.sdk.trace.SdkTracerProvider;
import io.opentelemetry.sdk.trace.export.ConsoleSpanExporter;
import io.opentelemetry.sdk.trace.export.SimpleSpanProcessor;
import io.opentelemetry.sdk.trace.samplers.Sampler;
import io.opentelemetry.sdk.resources.Resource;
import io.opentelemetry.semconv.ResourceAttributes;
import j*a.util.Map;
public class Main {
public static void main(String[] args) {
// 1. 配置 OpenTelemetry SDK
// 定义服务资源,例如服务名称
Resource serviceResource = Resource.getDefault()
.toBuilder()
.put(ResourceAttributes.SERVICE_NAME, "my-distributed-app")
.build();
// 配置 TracerProvider,使用 ConsoleSpanExporter 将 Span 输出到控制台
// 并设置采样器为始终采样
SdkTracerProvider tracerProvider = SdkTracerProvider.builder()
.addSpanProcessor(SimpleSpanProcessor.create(ConsoleSpanExporter.create()))
.setResource(serviceResource)
.setSampler(Sampler.alwaysOn())
.build();
// 构建并注册全局的 OpenTelemetry 实例
OpenTelemetry openTelemetry = OpenTelemetrySdk.builder()
.setTracerProvider(tracerProvider)
// OpenTelemetry SDK 会自动注册 W3CTraceContextPropagator 作为默认的传播器
// .setPropagators(ContextPropagators.create(W3CTraceContextPropagator.getInstance())) // 显式设置传播器
.buildAndRegisterGlobal();
// 2. 初始化调用方和被调用方服务
CallerService caller = new CallerService(openTelemetry);
CalleeService callee = new CalleeService(openTelemetry);
// 3. 模拟一次分布式调用
System.out.println("--- 模拟远程调用开始 ---");
Map<String, String> propagatedContext = caller.makeRemoteCall(); // 调用方注入上下文
callee.processRemoteCall(propagatedContext); // 被调用方提取上下文并创建子 Span
System.out.println("--- 远程调用模拟结束 ---");
// 4. 关闭 TracerProvider 以确保所有 Span 都被导出
tracerProvider.shutdown();
}
}运行此 Main 类,您将在控制台看到类似以下的输出(具体 Span ID 和 Trace ID 会有所不同):
--- 模拟远程调用开始 ---
调用方 Span ID: 6b3c...
注入的追踪上下文: {traceparent=00-d8b4...-6b3c...-01}
被调用方 Span ID: 7a8b...
被调用方父 Span ID: 6b3c...
--- 远程调用模拟结束 ---并且,ConsoleSpanExporter 会将完整的 Span 数据打印出来,显示 childSpanInCallee 的 parentSpanId 正是 parentSpanInCaller 的 spanId,证明父子关系已成功建立。
注意事项与最佳实践
- 理解 SpanContext 与 Span 对象的区别
以上就是OpenTelemetry J*a:利用上下文传播构建分布式 Span 关系的详细内容,更多请关注其它相关文章!
# java
# 是一个
# 河南seo怎么优化报价
# 卡车网站建设海报图
# 整合营销传播推广工具
# 仲恺seo推广优化价格
# SEO北京酒店推荐
# 丹东seo培训公司
# 怎么营销推广家居
# 长沙seo外链优化培训
# 灵宝网站建设模板
# 上饶网站推广 嶶心hfqjwl广告稳定
# 抽象类
# 多态
# 序列化
# 并将其
# 表现形式
# 如何使用
# 您需要
# 键值
# gate
# 键值对
# 常见问题
# 区别
# 注册表
# ai
# app
相关栏目:
【
科技资讯46185 】
【
网络学院92790 】
相关推荐:
AO3最新可访问网址 Archive of Our Own官方在线入口
React Hooks最佳实践:动态组件状态管理的组件化方案
快速CSGO开箱网站指南 CSGO开箱平台推荐
格力空气能E5故障代码是什么情况_格力空气能E5代码解析与应对措施
LINUX的I/O重定向是什么_深入理解LINUX中 >、>> 与 < 的区别
2025-2030年全球乘用车销量预测:新能源成增长主力
Descript怎样用AI剪辑自动去噪_Descript用AI剪辑自动去噪【自动降噪】
在React函数组件中利用原生HTML5进行邮箱地址验证
LINUX怎么设置定时任务_LINUX crontab配置教程
腾讯QQ邮箱官方网站_QQ邮箱网页版在线登录
微信聊天记录怎么加密_微信聊天记录加密方法
Go语言中高效处理x-www-form-urlencoded表单数据
蛙漫漫画免费阅读入口_蛙漫官方正版无广告纯净版
LINUX的perf命令入门_LINUX官方性能分析工具的使用与解读
MAC的“快捷指令”怎么同步到iPhone_MAC利用iCloud同步所有设备的自动化指令
黑猫投诉统一入口官网 消费者权益保护投诉平台
Golang如何实现Web文件静态资源服务器_Golang静态资源服务器开发与实践
小猿搜题在线学习页面在哪_小猿搜题在线学习中心入口
Lar*el DB::listen 事件中的查询执行时间单位解析
b站怎么取消点赞_b站点赞取消操作方法
钉钉视频会议声音异常如何处理 钉钉会议音频修复技巧
《刺客信条4:黑旗》重制版新细节曝光:无缝加载 地图更细致!
qq邮箱日历功能怎么用_创建日程与会议邀请的技巧
消息称三星明年 2 月正式发布 HBM4,与 SK 海力士同台竞技
在Socket.IO连接中实现Access Token自动更新与动态重连
淘宝网网页版登录入口 淘宝官方网页版快捷登录
初次安装JDK时环境变量如何正确配置_J*A_HOME与PATH设置规则讲解
高德地图沿途添加点失败如何解决 高德多点规划方法
Go语言中Map值调用指针接收器方法的限制与应对
为什么我的微信朋友圈看不到别人的更新_微信朋友圈更新显示异常解决方法
sublime侧边栏怎么增强功能_SideBarEnhancements for sublime安装与配置
今日头条怎么同步内容到抖音_今日头条内容同步到抖音教程
网易大神账号申诉需要多久_网易大神账号申诉流程说明
126邮箱手机版登录官网2026_126手机邮箱免费入口最新
外媒分析《GTA6》定价:卖100美元可以但真没必要!
vivo手机互传视频怎么操作_vivo手机互传视频详细传输方法
内存检查:在VS Code中调试C++时的内存视图
J*a TimerTask文件监控:HashMap状态管理与常见陷阱规避指南
韩剧圈正版入口页面_韩剧圈官网登录链接
J*aScript实现单选按钮与关联输入框的联动禁用教程
夸克AO3官网入口_AO3镜像网站2025推荐
如何使用J*aScript精确选择并批量修改特定父元素下子链接的样式
Win11 BitLocker密码忘了怎么办 Win11找回BitLocker恢复密钥方法【解决】
html5 app怎么运行环境_配html5 app运行环境【教程】
QQ邮箱登录首页官网地址2026 QQ邮箱官方网页入口
葱吃多了会怎样 葱吃多了会伤胃吗
手机屏幕碎了但能正常使用怎么办 手机外屏碎裂的修复建议
解决macOS上安装pyhdf时‘hdf.h’文件缺失的编译错误
vivo云服务网页版登录 怎么登录vivo云服务网页版
神庙逃亡小游戏在线玩 神庙逃亡小游戏入口


2025-12-05
浏览次数:次
返回列表
te final OpenTelemetry openTelemetry;
public CalleeService(OpenTelemetry openTelemetry) {
this.openTelemetry = openTelemetry;
this.tracer = openTelemetry.getTracer("my-callee-service");
}
/**
* 模拟处理远程调用,从传输载体中提取上下文,并创建子 Span。
* @param carrier 包含追踪上下文的 Map,模拟 HTTP Headers。
*/
public void processRemoteCall(Map<String, String> carrier) {
// 从 carrier 中提取追踪上下文
Context extractedContext = W3CTraceContextPropagator.getInstance().extract(
Context.current(), // 默认上下文,如果 carrier 中没有追踪信息,则使用此上下文
carrier, // 传输载体
new TextMapGetter<Map<String, String>>() {
@Override
public Iterable<String> keys(Map<String, String> carrier) {
return carrier.keySet();
}
@Override
public String get(Map<String, String> carrier, String key) {
return carrier.get(key);
}
});
// 使用提取到的上下文作为新 Span 的父级
Span childSpan = tracer.spanBuilder("childSpanInCallee")
.setParent(extractedContext) // 将提取到的上下文设置为父级
.startSpan();
try {
System.out.println("被调用方 Span ID: " + childSpan.getSpanContext().getSpanId());
System.out.println("被调用方父 Span ID: " + childSpan.getParentSpanContext().getSpanId());
// 模拟一些工作
Thread.sleep(100);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
} finally {
childSpan.end();
}
}
}