当前位置: 首页 > news >正文

《TCP/IP网络编程》学习笔记 | Chapter 19:Windows 平台下线程的使用

《TCP/IP网络编程》学习笔记 | Chapter 19:Windows 平台下线程的使用

  • 《TCP/IP网络编程》学习笔记 | Chapter 19:Windows 平台下线程的使用
    • 内核对象
      • 内核对象的定义
      • 内核对象归操作系统所有
    • 基于 Windows 的线程创建
      • 进程与线程的关系
      • Windows 中线程的创建方法
      • Windows 线程的销毁时间点
      • 编写多线程程序的环境设置
      • 创建“使用线程安全标准 C 函数”的线程
      • 句柄、内核对象和 ID 间的关系
    • 习题
      • (1)

《TCP/IP网络编程》学习笔记 | Chapter 19:Windows 平台下线程的使用

内核对象

要想掌握 Windows 平台下的线程,应首先理解内核对象(Kernel Objects)的概念。

内核对象的定义

操作系统创建的资源有很多种,如进程、线程、文件及即将介绍的信号量、互斥量等。其中大部分都是通过程序员的请求创建的,而且请求方式各不相同。虽然存在一些差异,但它们之间也有如下共同点:都是由 Windows 操作系统创建并管理的资源。

不同资源类型在管理方式也有差异。例如,文件管理中应注册并更新文件相关的数据I/O位置、文件的打开模式(rcad or write)等。如果是线程,则应注册并维护线程ID、线程所属进程等信息。操作系统为了以记录相关信息的方式管理各种资源,在其内部生成数据块(结构体变量)。当然,每种资源需要维护的信息不同,所以每种资源拥有的数据块格式也有差异。这类数据块称为“内核对象”。

假设在 Windows 下创建了 mydata.txt 文件,此时 Windows 操作系统将生成 1 个数据块以便管理,该数据块就是内核对象。同理,Windows 在创建进程、线程、线程同步信号量时也会生成相应的内核对象,用于管理操作系统资源。

内核对象归操作系统所有

线程、文件等资源的创建请求均在进程内部完成,因此,很容易产生“此时创建的内核对象所有者就是进程”的错觉。其实,内核对象所有者是内核(操作系统),内核对象的创建、管理、销毁时机的决定等工作均由操作系统完成。

基于 Windows 的线程创建

进程与线程的关系

现代操作系统都支持线程,因此,非显式创建线程的程序可描述为“单一线程模型的应用程序”,显式创建单独线程的程序可描述为“多线程模型的应用程序”。这就意味着 main 函数的运行同样基于线程完成,此时进程可以比喻为装有线程的篮子,实际的运行主体是线程。

Windows 中线程的创建方法

#include <windows.h>

HANDLE CreateThread(
	LPSECURITY_ATTRIBUTES lpThreadAttributes,
	SIZE_T dwStackSize,
	LPTHREAD_START_ROUTINE lpStartAddress,
	LPVOID lpParameter,
	DWORD dwCreationFlags,
	LPDWORD lpThreadId
);

参数:

  • IpThreadAttributes:线程安全相关信息,使用默认设置时传递 NULL。
  • dwStackSize:要分配给线程的栈大小,传递 0 时生成默认大小的栈。
  • IpStartAddress:传递线程的 main 函数信息。
  • lpParameter:调用 main 函数时传递的参数信息。
  • dwCreationFlags:用于指定线程创建后的行为,传递 0 时,线程创建后立即进入可执行状态。
  • IpThreadld:用于保存线程 ID 的变量地址值。

成功时返回线程句柄,失败时返回 NULL。

调用该函数将创建线程,操作系统为了管理这些资源也将同时创建内核对象。最后返回用于区分内核对象的整数型“句柄”(Handle)。

第 1 章已介绍过,句柄相当于 Linux的文件描述符。

上述定义看起来有些复杂,其实只需要考虑 IpStartAddress 和 lpParameter 这 2 个参数,剩下的只需传递 0 或 NULL 即可。

Windows 线程的销毁时间点

Windows 线程在首次调用的线程 main 函数返回时销毁(销毁时间点和销毁方法与 Linux 不同)。

还有其他方法可以终止线程,但最好的方法就是让线程main函数终止(返回)。

编写多线程程序的环境设置

旧的 VC++6.0版默认只包含支持单线程的库,需要自行配置。

在项目的属性-代码生成界面,可以检查运行库:

在这里插入图片描述

创建“使用线程安全标准 C 函数”的线程

通过 CreateThread 函数调用创建出的线程在使用 C/C++ 标准函数时并不稳定。

如果线程要调用 C/C++ 标准函数,需要通过如下方法创建线程:

#include <process.h>

unitptr_t _beginthreadex(
	void *security,
	unsigned stack_size,
	unsigned (*start_address)(void *),
	void *arglist,
	unsigned initflag,
	unsigned *thrdaddr
);

上述函数与之前的 CreateThread 函数相比,参数个数及各参数的含义和顺序的相同,只是变量名和参数类型有所不同。因此,用上述函数替换 CreateThread 函数时,只需适当更改数据类型。上述函数的返回值类型 uintptr_t 是 64 位 unsigned 整数型。

程序示例:

在这里插入代码片

运行结果:

在这里插入图片描述

与 Linux 相同,Windows 同样在 main 函数返回后终止进程,也同时终止其中包含的所有线程。另外,如果对上述代码进行运行的话,最后输出的内容并非字符串"end of main",而是"running thread"。但这是在 main 函数返回后,完全销毁进程前输出的字符串。

句柄、内核对象和 ID 间的关系

线程属于操作系统管理资源,伴随内核对象的创建,为了引用内核对象而返回句柄。

可以通过句柄区分内核对象,通过内核对象可以区分线程,最终,线程句柄成为区分线程的工具。

句柄的整数值在不同进程中可能重复。通过 CreateThread/_beginthreadex 函数可以得到线程 ID,它用于区分操作系统创建的所有线程,在跨进程范围内不会重复。

习题

(1)

相关文章:

  • 【Java项目】基于JSP的智能停车场管理系统
  • 蓝桥杯刷题 Day2 AC自动机(二次加强版)
  • linux 命令 vim
  • 若依前端框架增删改查
  • c++领域展开第十七幕——STL(vector容器的模拟实现以及迭代器失效问题)超详细!!!!
  • 三个线程按顺序交替打印 A B C
  • Power Apps 技术分享:使用控件的相对布局
  • 组态王Kingview配置为OPCUA服务器的一些问题处理
  • [快乐学坊management_1] With Cursor | Mysql设计 | 服务接口设计与开发
  • PyTorch 深度学习实战(17):Asynchronous Advantage Actor-Critic (A3C) 算法与并行训练
  • ABeam 德硕 | 在华外企ESG议题选择指南(5)—— 国际与国内ESG议题选择研究:SASB可持续会计准则解读
  • Unity 云渲染本地部署方案
  • 检查 YAML 文件格式是否正确的命令 yamllint
  • Vala编程语言教程-数据类型
  • 【iOS】SwiftUI 路由管理(NavigationStack)
  • 深入理解 Linux 的 top 命令:实时监控系统性能
  • Unity 解决TMP_Text 文字显示异常的问题
  • 手势调控屏幕亮度:Python + OpenCV + Mediapipe 打造智能交互体验
  • 记事本(基于JAVAGUI界面)
  • 一次模拟Windows挖矿病毒应急响应的流程及思路
  • 马上评|清理“滥竽充数者”,为医者正名
  • 刘强东坐镇京东一线:管理层培训1800人次,最注重用户体验
  • 沪指跌0.68%报3380.82点,创指跌1.92%:券商、军工跌幅靠前
  • 陕西省安康市汉阴县县长陈永乐已任汉阴县委书记
  • 上海市国防动员办公室副主任吴斌接受审查调查
  • 美国务卿鲁比奥将前往土耳其参加俄乌会谈