go-issues#14592 runtime: let idle OS threads exit 内核线程暴增与线程回收问题

简介: go-issues#14592 runtime: let idle OS threads exit 内核线程暴增与线程回收问题

前言

  在社区issues#14592可以看到,go中的空闲线程是不会自动回收的(注意是线程而不是协程,GMP模型中的M),那么就衍生出了三个问题

  1. 为什么会产生空闲线程
  2. 如何限制最大线程数量
  3. 怎么回收空闲线程

CPU线程与OS线程

这个一级目录算是补充一下os的知识吧

CPU线程与OS线程有什么区别和联系呢?操作系统中的进程可以很多,进程中的线程就更多了,常常有几十个上百个。OS中的线程有自己的栈空间,和同一进程中的其他线程共享地址空间等等;CPU中的线程就那么固定几个(例如四核八线程),是真正的计算资源。

两者都叫线程(Thread)是因为他们都是调度的基本单位。软件的调度基本单位是OS的线程,硬件的调度基本单位是CPU中的线程。操作系统负责把它产生的软线程调度到CPU中的硬线程中去进行计算

  GMP模型中的M皆是OS中的线程,所以后续所说的线程线程都是OS中的

为什么会产生空闲线程

  如果不熟悉GMP模型请先弄明白了深入理解GMP模型再接着读后续

  我们都知道GOMAXPROCS可以设置GMP中P的数量,那么GOMAXPROCS到底代表什么含义呢?

The GOMAXPROCS variable limits the number of operating system threads that can execute user-level Go code simultaneously.

There is no limit to the number of threads that can be blocked in system calls on behalf of Go code; those do not count against the GOMAXPROCS limit.

This package’s GOMAXPROCS function queries and changes the limit.

GOMAXPROCS 变量限制了可以同时执行用户级 Go 代码的操作系统线程数。

代表Go代码在系统调用中可以阻塞的线程数没有限制; 这些不计入GOMAXPROCS 限制。

这个包的 GOMAXPROCS 函数查询和更改限制。  注意这里的重点在系统调用中可以阻塞的线程数没有限制; 这些不计入GOMAXPROCS 限制,也就是说在系统调用中被阻塞的线程不在此限制之中。

 那么问题就来了,在深入理解GMP模型的场景10中,我们说过,一旦这个G陷入系统调用了,那么与之对应的M和P就会解除绑定,而G和M绑定在一起,直到解除阻塞,

 那么在陷入系统调用之后,因为在系统调用中可以阻塞的线程数没有限制; 这些不计入GOMAXPROCS 限制的原因,就会有新的M与P绑定,那么解除阻塞后,G回到了全局队列中,这个M呢?因为它没有P,所以它得不到任务,此时M就是空闲线程了。

 那么问题来了,如果短时间内,GO程序存在大量的系统调用,那线程数量不就暴涨了?

 这也就是为什么在深入理解GMP模型的(1)GMP模型简介里面说,GO语言本身的原因,把M限制为10000的原因,这个值存在的主要目的是限制可以创建无限数量线程的GO程序,即在程序把操作系统干爆之前,干掉程序。

如何限制最大线程数量

GO提供了debug.SetMaxThreads()方法可以让我们限制最大线程数量

先来看不做限制的情况

package main
import (
  "fmt"
  "net"
  "runtime/pprof"
  "sync"
)
var threadProfile = pprof.Lookup("threadcreate")
func main() {
  fmt.Println("创建协程之前的线程数量:", threadProfile.Count())
  var wg sync.WaitGroup
  for i := 0; i < 100; i++ {
    wg.Add(1)
    go func() {
      defer wg.Done()
      for j := 0; j < 100; j++ {
        net.LookupHost("www.baidu.com")
      }
    }()
  }
  wg.Wait()
  fmt.Println("创建协程之后的线程数量:", threadProfile.Count())
}
创建协程之前的线程数量: 6
创建协程之后的线程数量: 88

限制的情况

package main
import (
  "fmt"
  "net"
  "runtime/debug"
  "runtime/pprof"
  "sync"
)
var threadProfile = pprof.Lookup("threadcreate")
func main() {
  debug.SetMaxThreads(10)
  fmt.Println("创建协程之前的线程数量:", threadProfile.Count())
  var wg sync.WaitGroup
  for i := 0; i < 100; i++ {
    wg.Add(1)
    go func() {
      defer wg.Done()
      for j := 0; j < 100; j++ {
        net.LookupHost("www.baidu.com")
      }
    }()
  }
  wg.Wait()
  fmt.Println("创建协程之后的线程数量:", threadProfile.Count())
}
创建协程之前的线程数量: 6
runtime: program exceeds 10-thread limit
fatal error: thread exhaustion

当程序启动的线程M超过我们所设置的线程数量时,就会立马报错

怎么回收空闲线程

  issues#14592提出的问题就是让空闲的 OS 线程退出,不过目前并没有一个完美的解决方案,有人提出用runtime.LockOSThread()杀死线程,下面来介绍这个方法

  1. 调用 LockOSThread 函数会把当前 G 绑定在当前的系统线程 M 上,这个 G 总是在这个 M 上执行,并且阻止其它 G 在该 M 执行
  2. 只有当前 G 调用了与之前调用 LockOSThread 相同次数的 UnlockOSThread 函数之后,G 与 M 才会解绑
  3. 如果当前 G 在退出时,没有调用 UnlockOSThread,这个线程会被终止

利用第三个特性,在启动 G 时,调用 LockOSThread 来独占一个 M。当 G 退出时,而不调用 UnlockOSThread,那这个 M 就会被终止杀死了

package main
import (
  "fmt"
  "net"
  "runtime"
  "runtime/pprof"
  "sync"
)
var threadProfile = pprof.Lookup("threadcreate")
func main() {
  fmt.Println("创建协程之前的线程数量:", threadProfile.Count())
  var wg sync.WaitGroup
  for i := 0; i < 100; i++ {
    wg.Add(1)
    go func() {
      runtime.LockOSThread()
      defer wg.Done()
      for j := 0; j < 100; j++ {
        net.LookupHost("www.baidu.com")
      }
    }()
  }
  wg.Wait()
  fmt.Println("创建协程之后的线程数量:", threadProfile.Count())
}
# 现在的
创建协程之前的线程数量: 6
创建协程之后的线程数量: 13
# 之前的
创建协程之前的线程数量: 6
创建协程之后的线程数量: 88

由于调用了 runtime.LockOSThread 函数的 G 没有执行 UnlockOSThread 函数,在 G 执行完毕后,M 也被终止了,空闲线程大量减少了

不过这个方法其实是存在隐患的,具体看issues,下面是截取的片段

Hi @superajun-wsj Not sure that it is good solution. When the child process is created by one thread called A with PdeathSignal: SIGKILL and the thread A becomes idle, if the thread A exits, the child process will receive KILL signal. So I think UnLockOSThread might introduce other issues. just my two cents.

总结

  在绝大多数情况下,我们的程序并不会遇到空闲线程数过多的问题。如果真的存在线程数暴涨的问题,那么你应该思考代码逻辑是否合理(为什么你能允许短时间内如此多的系统同步调用),是否可以做一些例如限流之类的处理。而不是想着通过 SetMaxThreads 方法来处理。


目录
相关文章
|
4月前
|
存储 调度
探索操作系统的心脏:内核与用户空间的交互
在数字世界的每一次呼吸中,操作系统扮演着至关重要的角色。本文将深入探讨操作系统的核心组件——内核与用户空间之间的神秘舞蹈。通过直观的比喻和生动的代码片段,我们将一窥这场幕后的交响曲,了解它们是如何协同工作以支持我们的计算需求的。从简单的文件读写到复杂的网络通信,每一个操作背后都隐藏着内核与用户空间之间精妙的互动。准备好跟随我们的脚步,一起揭开操作系统的神秘面纱。
50 3
|
4月前
|
缓存 并行计算 Linux
深入解析Linux操作系统的内核优化策略
本文旨在探讨Linux操作系统内核的优化策略,包括内核参数调整、内存管理、CPU调度以及文件系统性能提升等方面。通过对这些关键领域的分析,我们可以理解如何有效地提高Linux系统的性能和稳定性,从而为用户提供更加流畅和高效的计算体验。
117 17
|
3月前
|
安全 Linux 开发者
探索操作系统的心脏:内核与用户空间的交互
在数字世界的每一次点击和命令背后,隐藏着一个复杂而精妙的操作系统世界。本文将带你走进这个世界的核心,揭示内核与用户空间的神秘交互。通过深入浅出的解释和直观的代码示例,我们将一起理解操作系统如何协调硬件资源,管理进程和内存,以及提供文件系统服务。无论你是编程新手还是资深开发者,这篇文章都将为你打开一扇通往操作系统深层原理的大门。让我们一起开始这段旅程,探索那些支撑我们日常数字生活的技术基石吧!
76 6
|
3月前
|
存储 缓存 网络协议
Linux操作系统的内核优化与性能调优####
本文深入探讨了Linux操作系统内核的优化策略与性能调优方法,旨在为系统管理员和高级用户提供一套实用的指南。通过分析内核参数调整、文件系统选择、内存管理及网络配置等关键方面,本文揭示了如何有效提升Linux系统的稳定性和运行效率。不同于常规摘要仅概述内容的做法,本摘要直接指出文章的核心价值——提供具体可行的优化措施,助力读者实现系统性能的飞跃。 ####
|
3月前
|
缓存 监控 网络协议
Linux操作系统的内核优化与实践####
本文旨在探讨Linux操作系统内核的优化策略与实际应用案例,深入分析内核参数调优、编译选项配置及实时性能监控的方法。通过具体实例讲解如何根据不同应用场景调整内核设置,以提升系统性能和稳定性,为系统管理员和技术爱好者提供实用的优化指南。 ####
|
4月前
|
存储 Linux 开发者
探索操作系统的内核——从理论到实践
操作系统是计算机科学的核心,它像一位默默无闻的指挥官,协调着硬件和软件之间的复杂关系。本文将深入操作系统的心脏——内核,通过直观的解释和丰富的代码示例,揭示其神秘面纱。我们将一起学习进程管理、内存分配、文件系统等关键概念,并通过实际代码,体验内核编程的魅力。无论你是初学者还是有经验的开发者,这篇文章都将带给你新的视角和知识。
|
3月前
|
机器学习/深度学习 人工智能 物联网
操作系统的心脏——深入理解内核机制
在本文中,我们揭开操作系统内核的神秘面纱,探索其作为计算机系统核心的重要性。通过详细分析内核的基本功能、类型以及它如何管理硬件资源和软件进程,我们将了解内核是如何成为现代计算不可或缺的基础。此外,我们还会探讨内核设计的挑战和未来趋势,为读者提供一个全面的内核知识框架。
|
3月前
|
消息中间件 安全 Linux
深入探索Linux操作系统的内核机制
本文旨在为读者提供一个关于Linux操作系统内核机制的全面解析。通过探讨Linux内核的设计哲学、核心组件、以及其如何高效地管理硬件资源和系统操作,本文揭示了Linux之所以成为众多开发者和组织首选操作系统的原因。不同于常规摘要,此处我们不涉及具体代码或技术细节,而是从宏观的角度审视Linux内核的架构和功能,为对Linux感兴趣的读者提供一个高层次的理解框架。
|
4月前
|
存储 调度 开发者
探索操作系统的心脏:内核与用户空间的交互之旅
在数字世界的无限广阔中,操作系统扮演着枢纽的角色,连接硬件与软件,支撑起整个计算生态。本篇文章将带领读者深入操作系统的核心——内核,揭示其与用户空间的神秘交互。我们将透过生动的例子和易于理解的比喻,深入浅出地探讨这一复杂主题,旨在为非专业读者揭开操作系统的神秘面纱,同时为有一定基础的读者提供更深层次的认识。从进程管理到内存分配,从文件系统到设备驱动,每一个环节都是精确而优雅的舞蹈,它们共同编织出稳定而高效的计算体验。让我们开始这场奇妙之旅,一探操作系统背后的科学与艺术。
46 5
|
4月前
|
缓存 网络协议 Linux
深入探索Linux操作系统的内核优化策略####
本文旨在探讨Linux操作系统内核的优化方法,通过分析当前主流的几种内核优化技术,结合具体案例,阐述如何有效提升系统性能与稳定性。文章首先概述了Linux内核的基本结构,随后详细解析了内核优化的必要性及常用手段,包括编译优化、内核参数调整、内存管理优化等,最后通过实例展示了这些优化技巧在实际场景中的应用效果,为读者提供了一套实用的Linux内核优化指南。 ####
115 1

热门文章

最新文章