原文地址:https://medium.com/a-journey-with-go/go-how-does-gops-interact-with-the-runtime-778d7f9d7c18

本文基于 Go 1.13 和 gops 0.3.7.

gops 旨在帮助开发人员诊断 Go 流程并与之交互。它提供了跟踪运行中的程序几秒钟的功能,可以通过获取 CPU 配置文件 pprof,甚至可以直接与垃圾收集器进行交互。

发现

gops 提供发现服务,该服务能够列出计算机上运行的 Go 进程。gops 不带参数运行仅显示 Go 进程。为了举例说明,我启动了一个程序,该程序可以计算高达一百万的素数。这是流程发现的输出:

295 1 gops          go1.13 /go/src/github.com/google/gops/gops
168 1 prime-number* go1.13 /go/prime-number/prime-number

gops 看到程序启动以及它自己的过程。基于此输出,我们唯一需要的是进程 ID 就可以开始与程序进行交互。但是,让我们了解 gops 是如何只过滤 Go 进程。

首先,gops 列出所有过程。然后,对于每个进程,它将打开二进制文件以读取其符号表:

如果符号表包含 runtime.main(主 goroutine 的入口)或 main.main(我们程序的入口),则可以将其标记为 Go 程序。

有关符号表的更多信息,建议您阅读我的文章 “Go:如何利用符号表”。要了解有关主 goroutine 的更多信息,建议您阅读我的文章 “ Go:g0,Special Goroutine”。

gops 也会通过之前的符号表里面的 runtime.buildVersion获取使用的 Go 版本。但是,由于可以从二进制文件中删除符号表,因此 gops 需要另一种方法来检测 Go 二进制文件。让我们用剥离后的二进制文件再试一次:

295 1 gops            go1.13             /go/src/..../gops
168 1 prime-number-s* unknown Go version /go/.../prime-number-s

如果程序正确地标记为 Go 二进制文件,则由于缺少符号表,因此无法再检测 Go 版本。根据该可执行文件格式 - ELF,MZ 等 - gops 读取寻找嵌入在二进制版本 ID 的部分。一旦发现完成,它就可以开始与程序进行交互。

交互

与其他 Go 程序进行交互的唯一条件是确保它们启动了 gops 代理。该代理是一个简单的侦听器,将为 gops 请求提供服务。只需添加以下行即可:

if err:= agent.Listen(agent.Options {}); err!= nil { log.Fatal(err)
}

然后,具有可用代理的任何程序都可以与 gops 进行交互。这是命令的示例 stats

# gops stats 168
goroutines: 6210
OS threads: 9
GOMAXPROCS: 2
num CPU: 2

有关更多命令,您可以参考项目文档。如果缺少该代理,则在与该代理进行交互时会收到错误消息:

Couldn't resolve addr or pid 168 to TCPAddress: couldn't get port for PID 168

该错误表明 gops 正在通过 TCP 寻找暴露的端点以便与程序进行通信。让我们画出软件包的工作流程以了解其工作原理。

工作流程

gops 与 Go 程序之间的通信是通过 TCP 和 Go 程序的暴露端点进行的:

分配给每个程序的端口都写在配置文件中,例如, path/to/config/{processID}可以很容易让 gops 知道暴露的端口。然后,gops 可以将命令标志发送到代理将在其中收集数据并进行响应的程序:

gops 是怎么和 Go 的运行时进行交互的?相关推荐

  1. 二进制包如何知道go 版本_gops 是怎么和 Go 的运行时进行交互的?

    本文基于 Go 1.13 和 gops 0.3.7. gops 旨在帮助开发人员诊断 Go 流程并与之交互.它提供了跟踪运行中的程序几秒钟的功能,可以通过获取 CPU 配置文件 pprof,甚至可以直 ...

  2. 嵌入式Linux设备驱动程序:在运行时读取驱动程序状态

    嵌入式Linux设备驱动程序:在运行时读取驱动程序状态 Embedded Linux device drivers: Reading driver state at runtime 在运行时了解驱动程 ...

  3. 编译器设计-RunTime运行时环境

    编译器设计-RunTime运行时环境 Compiler Design - Run-Time Environment 作为源代码的程序仅仅是文本(代码.语句等)的集合,要使其活动,它需要在目标计算机上执 ...

  4. java运行时_java编译时与运行时概念与实例详解

    Java编译时与运行时很重要的概念,但是一直没有明晰,这次专门博客写明白概念. 基础概念 编译时 编译时顾名思义就是正在编译的时候.那啥叫编译呢?就是编译器帮你把源代码翻译成机器能识别的代码.(当然只 ...

  5. oracle java rmi 漏洞,Oracle Java SE Java运行时环境RMI子组件远程漏洞(CVE-2011-3556)

    发布日期:2011-10-20 更新日期:2011-10-20 受影响系统: Oracle Sun JRE 1.6.x Oracle Sun JDK 1.6.x 不受影响系统: Oracle Sun ...

  6. Linux 下 进程运行时内部函数耗时的统计 工具:pstack,strace,perf trace,systemtap

    简单记录一些 在linux下 统计进程内部函数运行耗时的统计工具,主要是用作性能瓶颈分析.当然以下工具除了pstack功能单一之外,其他的工具都非常强大,这里仅仅整理特定场景的特定用法,用作协同分析. ...

  7. c# .netframwork 4.0 调用 2.0时报错 混合模式程序集是针对“v2.0.50727”版的运行时生成的,在没有配置其他信息的情况下,无法在 4.0 运行时中加载该程序集。...

    "System.IO.FileLoadException"类型的未经处理的异常在 XXX.dll 中发生 其他信息: 混合模式程序集是针对"v2.0.50727" ...

  8. VC运行时库(/MD、/MT等)

    VC项目属性→配置属性→C/C++→代码生成→运行时库 可以采用的方式有:多线程(/MT).多线程调试(/MTd).多线程DLL(/MD).多线程调试DLL(/MDd).单线程(/ML).单线程调试( ...

  9. Linux系统程序运行时加载动态库路径顺序

    程序运行时加载动态库路径顺序(Linux) 在linux系统中,如果程序需要加载动态库,它会按照一定的顺序(优先级)去查找: 链接时路径(Link-time path)和运行时路径(Run-time ...

最新文章

  1. device tree --- #address-cells and #size-cells property【转】
  2. 10元权限gm游戏_游戏P图超能打!揭秘10年老本儿500元升级计划
  3. Linux下守护进程(daemon)的实现
  4. ABAP 编程语言中的系统字段(System Fields)
  5. 学习进度(2016.3.13)
  6. Mapreduce中maptask过程详解
  7. 新rust怎么拆除围墙_“问题围挡”拆除 街道变漂亮了
  8. 这7本O’Reilly推出的免费Python电子书,够你看了
  9. RHEL7 -- 使用Chrony设置时间与时钟服务器同步
  10. Python基础知识笔记(二)
  11. 对c语言课程的收获,c语言课程设计心得体会精选.doc
  12. eclipse svn忽略指定文件或文件夹
  13. NATAPP 内网穿透的使用
  14. 在VS2010配置并运行PBC库程序
  15. 技术栈(technology stack)
  16. 转自【MDCC技术大咖秀】Android内存优化之OOM
  17. 在线编译器汇总|2020年最新版
  18. nas存储如何做远程服务器数据备份_NAS存储与数据备份方案
  19. 好莱坞大片! 为躲避死亡威胁, 只用15步, 这个密码朋克大叔就从世界消失了......
  20. 代码角度理解SGX的认证机制(一):本地认证

热门文章

  1. python——输入出生和月,计算出接下来的生日距离今天还有多少天
  2. DITA-OT发布过程和中文支持
  3. uharc 命令参数使用
  4. 录像机获取服务器信息失败,监控录像机找不到服务器
  5. 用ros做nat 共享上网
  6. 诺基亚手机信号服务器,诺基亚手机什么型号信号最强
  7. AI人脸自动签到识别系统使用教程
  8. siesta在Linux运行,[求助]Linux下安装Siesta的方法 - 计算模拟 - 小木虫 - 学术 科研 互动社区...
  9. 程序员拒绝PM小姐姐需求的 4 大理由!
  10. storm 开发实例