Win32进程句柄表与内核对象深度解析
在Win32进程中,句柄表与内核对象的关系至关重要。句柄表通过索引允许三环程序访问位于高地址空间的内核对象,如EPROCESS结构。内核对象包括进程、线程、事件等多种类型,可通过CloseHandle API进行管理。多进程可以通过OpenProcess API共享内核对象,引用计数机制确保对象在无引用时被销毁。进程间还可以通过继承句柄技术共享资源,具体由安全属性结构体中的bInheritHandle参数决定。此外,进程ID(PID)是全局句柄表的索引,理解这些概念有助于掌握Windows程序开发中的进程操作和资源管理。
在不改变文章大意和图片位置的前提下,以下是经过伪原创处理的文章:
句柄表与内核对象之间的关系是什么?首先,我们需要了解什么是句柄表,什么是内核对象。
一、句柄表和内核对象的概念
- 句柄表的生成
当我们使用CreateProcess函数时,它会返回一个进程句柄和一个线程句柄。在调用CreateProcess时,内核会创建一个EPROCESS结构来保存进程信息。
然而,如何让三环程序使用这个EPROCESS呢?直接返回EPROCESS是不行的,因为EPROCESS位于高两G的地址空间,三环程序无法访问。为了解决这个问题,Windows创建了一个表格,并返回这个表格的索引。我们使用的就是这个索引。
- 什么是内核对象
内核对象就是我们提到的EPROCESS。内核对象有很多种,具体可以参考CloseHandle API,它可以关闭哪些内核对象:
- Access token
- Communications device
- Console input
- Console screen buffer
- Event
- File
- File mapping
- I/O completion port
- Job
- Mailslot
- Memory resource notification
- Mutex
- Named pipe
- Pipe
- Process
- Semaphore
- Thread
- Transaction
- Waitable timer
这些对象可以操作事件、文件、互斥体、线程等。
二、多进程共享内核对象
- 第一种方法:使用OpenProcess
在Windows程序中,我们操作的都是内核对象。我们可以通过OpenProcess API来打开一个已有的进程的内核对象。
每个进程的句柄表都是私有的。例如,在第一张表中,句柄索引为1,对应的内核对象为A。如果将这个索引传给B进程,是没有用的。B进程只有在使用API打开后,才能获得A内核对象。
中间的紫色表代表引用计数。每次引用内核对象,这个值会加1。CloseHandle的作用是使内核对象的引用计数减1。如果所有引用都被关闭,那么内核对象将无人使用,也没有任何引用,因此会被销毁。也就是说,当内核对象的引用计数为0时,它才会被真正销毁。
线程是一个特例:当线程的内核对象引用计数为0时,它不会被关闭。此时,必须先关闭线程,然后使用CloseHandle使引用计数减1。
- 使用继承句柄技术
在Windows程序中,A进程创建B进程,或者带有内核对象的API在创建时,都有一个SD属性,即安全属性。这个属性可以表示你创建的句柄是否可以被继承。
例如,CreateEvent()创建事件。我们先不讨论API的作用,看看它的参数:
HANDLE CreateEventA( LPSECURITY_ATTRIBUTES lpEventAttributes, // 安全属性结构体 BOOL bManualReset, BOOL bInitialState, LPCSTR lpName);
第一个参数是安全属性结构体。如果我们不指定,默认使用父进程的。
安全属性结构体如下:
typedef struct _SECURITY_ATTRIBUTES { DWORD nLength; // 当前结构体大小,Windows扩展使用 LPVOID lpSecurityDescriptor; // 表明这个句柄给谁使用,谁可以访问,谁可以关闭。不重要,具体可以查看API中的结构体定义 BOOL bInheritHandle; // 重要,表明句柄是否可以被继承 } SECURITY_ATTRIBUTES, *PSECURITY_ATTRIBUTES, *LPSECURITY_ATTRIBUTES;
如果我们的句柄可以被继承,那么句柄表的第一项就会填1,表示这个句柄可以被继承。如果不能继承,则为0。
此时,子进程就可以继承父进程的所有可继承的句柄表。注意,是所有可继承的句柄,可以共享,如下图所示:
A进程创建的B和D是可以继承的,因此子进程可以完全复制A进程的可继承句柄表。不允许继承的句柄则赋值为0,如下图所示:
三、进程PID解析
在Windows任务管理器中,有PID选项,我们可以选择查看。而且在Windows中也经常听到进程ID的概念。
那么,进程ID到底是什么?
其实,进程ID是全局句柄表的一个索引。上面提到的句柄表都是私有的句柄表,而PID是全局句柄表中的。
这个全局句柄表记录了所有正在运行的进程的句柄,而且是唯一的。如果进程死亡,这个PID可能会指向其他句柄,但也是唯一的,如下图所示:
这个全局句柄表才是真正有意义的。为什么这么说呢?
我们可以做个测试:
- 使用OpenProcess打开进程句柄。
- 使用TerminateProcess结束进程。
OpenProcess(访问权限, 句柄是否可以继承, 进程PID) TerminateProcess(进程句柄, 自定义的退出码)
使用这两个API可以测试我们已有的进程是否可以被关闭。如果测试后你会发现,只有通过PID获得的句柄才是有用的,也就是说,全局句柄表才是关键,而上面提到的都是子进程的句柄表。
四、常用进程操作API
- GetModuleFileName():获取当前模块路径,例如:c:\1.exe
- GetCurrentDirectory():获取当前的工作目录,例如:c:\text\abc
- OpenProcess():根据进程PID打开进程,获取进程句柄。
- FindWindow():根据类名和文件名,返回窗口句柄。
- GetWindowThreadProcessId():根据窗口句柄,获取进程PID。
- EnumProcesses:遍历所有进程,返回进程PID。具体参考MSDN,有提供的例子。
- GetCommandLine():获取命令行参数。
- CreateToolhelp32Snapshot():创建进程快照。如果你了解逆向工程,你会知道FS寄存器中的TEB和PEB结构中存储了当前模块或进程的链表。这是保存当前时刻的快照。我们可以进行遍历,具体参考MSDN或本博客。
五、编写Windows程序遇到的问题
在编写Windows程序时,我们会包含windows.h,但有些函数可能没有。例如,我们提到的第八个函数,快照函数。
此时,我们需要查询MSDN。我们可以在网页上搜索一下:
我们可以在下方看到所需的头文件是tlhelp32.h,此时我们包含它即可。
遇到的问题2:
有时候我们包含了头文件并使用了,但调用API时出错。为什么?
原因是有些API在高版本中才有,低版本中使用时没有导出。此时使用会出错,提示没有这个API。
解决方法:如果你学过Win32,你会理解这个方法。如果没有学过也没关系。这种问题很少遇到,博主也只遇到过一次。
可以使用LoadLibrary加载所需的DLL,然后使用GetProcAddress获取函数地址,使用函数指针来调用这个函数。
以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持golang学习网!更多关于文章的相关知识,也可关注golang学习网公众号。

- 上一篇
- Debian上Tigervnc远程打印设置攻略

- 下一篇
- Python字典中value的详细解析及取值方法
-
- 文章 · 软件教程 | 11分钟前 |
- 电脑开机黑屏原因及解决方法
- 151浏览 收藏
-
- 文章 · 软件教程 | 17分钟前 |
- 电脑风扇转速异常解决方法
- 194浏览 收藏
-
- 文章 · 软件教程 | 27分钟前 |
- 电脑开机黑屏怎么办黑屏解决方法大全
- 378浏览 收藏
-
- 文章 · 软件教程 | 48分钟前 |
- Win10安装失败原因及解决方法
- 265浏览 收藏
-
- 文章 · 软件教程 | 1小时前 |
- Win11免密登录设置方法详解
- 110浏览 收藏
-
- 文章 · 软件教程 | 2小时前 |
- Win10玩红警打不开?兼容性设置教程
- 292浏览 收藏
-
- 文章 · 软件教程 | 2小时前 |
- 电脑死机重启原因及检测方法
- 460浏览 收藏
-
- 文章 · 软件教程 | 3小时前 |
- WindowsDefender误删文件怎么恢复
- 497浏览 收藏
-
- 文章 · 软件教程 | 3小时前 |
- 联想开机黑屏0x00000024解决方法
- 443浏览 收藏
-
- 文章 · 软件教程 | 3小时前 |
- Win8重装系统怎么按键?启动快捷键大全
- 324浏览 收藏
-
- 文章 · 软件教程 | 3小时前 |
- 光盘装Win8要多久?Win8安装时间解析
- 264浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 511次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 498次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- 畅图AI
- 探索畅图AI:领先的AI原生图表工具,告别绘图门槛。AI智能生成思维导图、流程图等多种图表,支持多模态解析、智能转换与高效团队协作。免费试用,提升效率!
- 27次使用
-
- TextIn智能文字识别平台
- TextIn智能文字识别平台,提供OCR、文档解析及NLP技术,实现文档采集、分类、信息抽取及智能审核全流程自动化。降低90%人工审核成本,提升企业效率。
- 33次使用
-
- 简篇AI排版
- SEO 简篇 AI 排版,一款强大的 AI 图文排版工具,3 秒生成专业文章。智能排版、AI 对话优化,支持工作汇报、家校通知等数百场景。会员畅享海量素材、专属客服,多格式导出,一键分享。
- 30次使用
-
- 小墨鹰AI快排
- SEO 小墨鹰 AI 快排,新媒体运营必备!30 秒自动完成公众号图文排版,更有 AI 写作助手、图片去水印等功能。海量素材模板,一键秒刷,提升运营效率!
- 29次使用
-
- Aifooler
- AI Fooler是一款免费在线AI音频处理工具,无需注册安装,即可快速实现人声分离、伴奏提取。适用于音乐编辑、视频制作、练唱素材等场景,提升音频创作效率。
- 35次使用
-
- pe系统下载好如何重装的具体教程
- 2023-05-01 501浏览
-
- qq游戏大厅怎么开启蓝钻提醒功能-qq游戏大厅开启蓝钻提醒功能教程
- 2023-04-29 501浏览
-
- 吉吉影音怎样播放网络视频 吉吉影音播放网络视频的操作步骤
- 2023-04-09 501浏览
-
- 腾讯会议怎么使用电脑音频 腾讯会议播放电脑音频的方法
- 2023-04-04 501浏览
-
- PPT制作图片滚动效果的简单方法
- 2023-04-26 501浏览