0

0

句柄泄露问题追踪

看不見的法師

看不見的法師

发布时间:2025-10-01 08:08:23

|

225人浏览过

|

来源于php中文网

原创

在编写windowslinux程序时,句柄泄露是一个常见的问题。在linux中,一个进程的文件描述符(fd)使用是有上限的,可以通过ulimit命令查看这个上限。当发生fd泄露时,可能会导致socket创建失败或文件无法打开等问题。windows也有类似的限制,本文主要介绍如何在windows中追踪句柄泄露的问题。

在Windows开发中,调用Windows API如CreateFileCreateEventCreateThread等时,会返回一个句柄Handle。如果在使用完资源后没有调用CloseHandle关闭Handle,就会出现句柄泄露的问题。当这种情况发生时,再次调用如CreateThread可能会返回Windows错误代码1450,表示“系统资源不足以完成请求的服务”,从而导致程序运行问题。Windows系统的总句柄数是有限制的,这甚至可能影响其他进程的运行。接下来,我们将探讨如何定位句柄泄露问题。

使用Process Explorer定位句柄泄露可以在任务管理器中查看进程的句柄数量,也可以在Process Explorer中查看。我们可以通过以下步骤来定位句柄泄露问题:

  1. Process Explorer中显示Handles列,如果进程存在句柄泄露问题,该列的数值会持续增长。
  2. 选中相应的进程,可以查看该进程的句柄详细信息,如句柄关联的线程、文件、Event等。
  3. 当发生句柄泄露时,会有大量相似类型的句柄出现。

如果是由于CreateThread的句柄未释放导致的句柄泄露,可以在句柄详细信息中看到许多Thread类型的句柄,然后查找可能调用CreateThread的代码。

如果是由于CreateFile的句柄未释放,可以在Process Explorer中查看文件路径,根据文件路径查找可能引起句柄泄露的代码。

句柄泄露问题追踪

这种方法可以解决部分句柄泄露问题,但有时可能遇到无法解决的情况:

  • 一个产品可能依赖多个第三方模块,如果句柄泄露是第三方模块引起的,可能难以通过泄露句柄的类型和名称定位到具体模块。
  • Process Explorer无法显示所有句柄,如无名的Event,导致无法查找。

使用Windbg定位句柄泄露问题除了上述问题,是否有方法可以定位到导致句柄泄露的具体代码位置?Windbg可以做到。我们先看一段测试代码,每隔一秒创建一个Event,但没有调用CloseHandle,这会导致句柄泄露。

Frase
Frase

Frase是一款出色的长篇 AI 写作工具,快速创建seo优化的内容。

下载
#include 
#include 
#include 

void HandleLeak() { int iCount = 0; while (true) { iCount++; HANDLE hEvent = CreateEvent(NULL, TRUE, FALSE, NULL); DWORD dwError = GetLastError(); std::this_thread::sleep_for(std::chrono::seconds(1)); if (!hEvent || dwError != ERROR_SUCCESS) { std::cerr << "Failed to create event. Error code: " << dwError << std::endl; break; } } }

int main() { std::thread t(HandleLeak); t.join(); return 0; }

第一步:使用Windbg附加到要测试的进程。

第二步:在Windbg中调用命令!htrace -enable,开启句柄追踪,并保存当前所有Handle的快照。

0:006> !htrace -enable
Handle tracing enabled.
Handle tracing information snapshot successfully taken.

第三步:在Windbg中调用命令g,让程序运行一段时间。

第四步:通过菜单Debug->Break进入调试状态,然后在Windbg中运行!htrace -diff,将进程当前的所有句柄与之前的快照进行对比,找出这段时间内多出来的句柄。

0:006> !htrace -diff
Handle tracing information snapshot successfully taken.
0x31 new stack traces since the previous snapshot.
Ignoring handles that were already closed...
Outstanding handles opened since the previous snapshot:

Handle = 0x0000000000000290 - OPEN Thread ID = 0x0000000000001ca0, Process ID = 0x0000000000004360 0x00007ffca4dcb2a4: ntdll!NtCreateEvent+0x0000000000000014 0x00007ffca1ebb623: KERNELBASE!CreateEventA+0x0000000000000083 0x00007ff7ea001e94: HandleLeak!HandleLeak+0x0000000000000034 0x00007ff7ea002099: HandleLeak!main+0x0000000000000009 0x00007ff7ea0023d4: HandleLeak!__scrt_common_main_seh+0x000000000000010c 0x00007ffca23f4034: KERNEL32!BaseThreadInitThunk+0x0000000000000014 0x00007ffca4da3691: ntdll!RtlUserThreadStart+0x0000000000000021

Handle = 0x0000000000000280 - OPEN Thread ID = 0x0000000000001ca0, Process ID = 0x0000000000004360 0x00007ffca4dcb2a4: ntdll!NtCreateEvent+0x0000000000000014 0x00007ffca1ebb623: KERNELBASE!CreateEventA+0x0000000000000083 0x00007ff7ea001e94: HandleLeak!HandleLeak+0x0000000000000034 0x00007ff7ea002099: HandleLeak!main+0x0000000000000009 0x00007ff7ea0023d4: HandleLeak!__scrt_common_main_seh+0x000000000000010c 0x00007ffca23f4034: KERNEL32!BaseThreadInitThunk+0x0000000000000014 0x00007ffca4da3691: ntdll!RtlUserThreadStart+0x0000000000000021

第五步:通过上述Handle调用,可以很容易地找到导致句柄泄露的代码。

相关专题

更多
java中break的作用
java中break的作用

本专题整合了java中break的用法教程,阅读专题下面的文章了解更多详细内容。

118

2025.10.15

java break和continue
java break和continue

本专题整合了java break和continue的区别相关内容,阅读专题下面的文章了解更多详细内容。

256

2025.10.24

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

392

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

572

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

481

2023.08.10

Java 并发编程高级实践
Java 并发编程高级实践

本专题深入讲解 Java 在高并发开发中的核心技术,涵盖线程模型、Thread 与 Runnable、Lock 与 synchronized、原子类、并发容器、线程池(Executor 框架)、阻塞队列、并发工具类(CountDownLatch、Semaphore)、以及高并发系统设计中的关键策略。通过实战案例帮助学习者全面掌握构建高性能并发应用的工程能力。

61

2025.12.01

windows查看端口占用情况
windows查看端口占用情况

Windows端口可以认为是计算机与外界通讯交流的出入口。逻辑意义上的端口一般是指TCP/IP协议中的端口,端口号的范围从0到65535,比如用于浏览网页服务的80端口,用于FTP服务的21端口等等。怎么查看windows端口占用情况呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

601

2023.07.26

查看端口占用情况windows
查看端口占用情况windows

端口占用是指与端口关联的软件占用端口而使得其他应用程序无法使用这些端口,端口占用问题是计算机系统编程领域的一个常见问题,端口占用的根本原因可能是操作系统的一些错误,服务器也可能会出现端口占用问题。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1104

2023.07.27

Java JVM 原理与性能调优实战
Java JVM 原理与性能调优实战

本专题系统讲解 Java 虚拟机(JVM)的核心工作原理与性能调优方法,包括 JVM 内存结构、对象创建与回收流程、垃圾回收器(Serial、CMS、G1、ZGC)对比分析、常见内存泄漏与性能瓶颈排查,以及 JVM 参数调优与监控工具(jstat、jmap、jvisualvm)的实战使用。通过真实案例,帮助学习者掌握 Java 应用在生产环境中的性能分析与优化能力。

0

2026.01.20

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 7.5万人学习

Excel 教程
Excel 教程

共162课时 | 12.5万人学习

PHP基础入门课程
PHP基础入门课程

共33课时 | 2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号