【Windows线程开发】线程基础

简介: 【Windows线程开发】线程基础

一.线程基本概念

Windows线程是可以执行的代码的实例,系统是以线程为单位调度程序。一个程序中可以有多个线程,实现多任务的处理。

  • Windows线程的特点:
  1. 每个线程都具有一个ID
  2. 每个线程都具有自己的内存栈
  3. 同一进程中的线程使用同一个地址空间
  • 线程的调度:
    将CPU的执行时间划分成时间片,依次根据时间片执行不同的线程
    线程轮询:线程A->线程B->线程A…

二.创建线程

HANDLE CreateThread(
  LPSECURITY_ATTRIBUTES lpThreadAttributes,//指向SECURITY_ATTRIBUTES结构的指针,该结构确定返回的句柄是否可以由子进程继承
  SIZE_T dwStackSize,//堆栈的初始化大小(以字节为单位)
  LPTHREAD_START_ROUTINE lpStartAddress,//指向由进程执行的应用程序定义函数的指针
  LPVOID lpParmeter,//指向要传递给线程的变脸指针
  DWORD dwCreationFlags,//控制线程创建的标志
  LPDWORD lpThreadID //指向接收线程标识(ID)的变量指针
);

参数说明:

上述参数为微软官方文档中的说法,可能有些不理解,这里我用通俗的语言来向大家解释:

lpThreadAttributes::安全属性

dwStackSize:线程栈的大小,这里的线程栈的大小是以1兆对齐的

lpStartAddress:线程处理函数的函数地址

lpParameter:传递给线程处理函数的参数(这里我们想传给线程处理函数什么参数,我就就可以填什么参数,但是要注意类型的转换

dwCreationFlags:线程的创建方式(立即创建或挂起等)

lpThreadID:这里我们只需要填上变量的地址,该函数会自动填入线程ID

返回值:如果创建成功,则返回线程句柄

DWORD WINAPI ThreadProc(
  LPVOID lpParameter   //创建线程事,传递给线程处理函数的参数
);

三.线程实例(单线程,多线程)

单线程执行

我们来通过一段代码来看看单线程执行:

// 线程.cpp : 此文件包含 "main" 函数。程序执行将在此处开始并结束。
//
#include <iostream>
#include <windows.h>
using namespace std;
DWORD WINAPI TreadProc(LPVOID lpParameter);
int main()
{
    //创建线程
    char a[] = "1111111111111111111";
    int ID1 = 0;
    HANDLE hTread1 = CreateThread(NULL, 1, TreadProc, a, 0, (LPDWORD)ID1);
    return 0;
}
DWORD WINAPI TreadProc(LPVOID lpParameter) {
    while (1) {
        cout << lpParameter << endl;
        Sleep(1000);
    }
}

我们来执行这一段代码:

发现出现了错误,线程处理函数根本没有执行,或者说我们根本没有看到。

我们来分析一下问题所在:

我们不难发现,在创建了进程之后,主进程退出了,所以说我们创建的线程也退出了,我们可以使用一个阻塞函数getchar()来观察:

// 线程.cpp : 此文件包含 "main" 函数。程序执行将在此处开始并结束。
//
#include <iostream>
#include <windows.h>
using namespace std;
DWORD WINAPI TreadProc(LPVOID lpParameter);
int main()
{
    //创建线程
    char a[] = "Hello World";
    int ID1 = 0;
    HANDLE hTread1 = CreateThread(NULL, 1, TreadProc, a, 0, (LPDWORD)ID1);
    getchar();
    return 0;
}
DWORD WINAPI TreadProc(LPVOID lpParameter) {
    char* a = (char*)lpParameter;
    while (1) {
        cout << a << endl;
        Sleep(1000);
    }
}

这样我们就能看到单线程的执行了:

多线程执行

我们创建多个线程来看看多线程的执行:

// 线程.cpp : 此文件包含 "main" 函数。程序执行将在此处开始并结束。
//
#include <iostream>
#include <windows.h>
using namespace std;
DWORD WINAPI TreadProc(LPVOID lpParameter);
int main()
{
    //创建线程
    char a[] = "111111111111";
    int ID1 = 0;
    HANDLE hTread1 = CreateThread(NULL, 1, TreadProc, a, 0, (LPDWORD)ID1);
    char b[] = "222222222222";
    int ID2 = 0;
    HANDLE hTread2 = CreateThread(NULL, 1, TreadProc, b, 0, (LPDWORD)ID2);
    char c[] = "333333333333";
    int ID3 = 0;
    HANDLE hTread3 = CreateThread(NULL, 1, TreadProc, c, 0, (LPDWORD)ID3);
    getchar();
    return 0;
}
DWORD WINAPI TreadProc(LPVOID lpParameter) {
    char* a = (char*)lpParameter;
    while (1) {
        cout << a << endl;
        Sleep(1000);
    }
}

我们来看看执行效果:

四.挂起,销毁线程

  • 挂起线程
    我们在创建线程的时候,可以设置创建方式为CREATE_SUSPENDED也可以挂起线程
DWORD SuspendTread(
  HANDLE hThread    //线程句柄
);
  • 唤醒线程
DWORD ResumeThread(
  HANDLE hThread   //线程句柄
);
  • 结束指定线程
BOOL TerminateThread(
  HANDLE hThread,        //线程句柄
  DWORD dwExitCode       //退出代码
);
  • 结束函数所在线程
VOID ExitThread(
  DWORD dwExitCode       //退出代码
);

我们来写一个简单的双线程,一个处于挂起状态,一个处于执行状态,当跳过getchar() 函数后,交换两个线程的状态,以此来展示挂起线程和结束线程:

// 线程.cpp : 此文件包含 "main" 函数。程序执行将在此处开始并结束。
//
#include <iostream>
#include <windows.h>
using namespace std;
DWORD WINAPI TreadProc(LPVOID lpParameter);
int main()
{
    //创建线程
    char a[] = "111111111111";
    int ID1 = 0;
    HANDLE hThread1 = CreateThread(NULL, 1, TreadProc, a, 0, (LPDWORD)ID1);
    char b[] = "222222222222";
    int ID2 = 0;
    HANDLE hThread2 = CreateThread(NULL, 1, TreadProc, b,CREATE_SUSPENDED, (LPDWORD)ID2);
    getchar();
    SuspendThread(hThread1);
    ResumeThread(hThread2);
    getchar();
    return 0;
}
DWORD WINAPI TreadProc(LPVOID lpParameter) {
    char* a = (char*)lpParameter;
    while (1) {
        cout << a << endl;
        Sleep(1000);
    }
}

我们来看看执行效果:

五.线程相关操作

  • 获取当前线程ID
    GetCurrentThreadID()函数
  • 获取当前线程句柄
    GetCurrentThread()函数
  • 等候单个句柄有信号
    这里介绍一下可等候句柄:只有当一个句柄有有信号和无信号两种状态时,才可以称为可等候信号
    例:线程句柄:当线程执行时,为无信号状态,当线程结束时,为有信号状态
VOID WaitForSingleObject(
  HANDLE handle,             //句柄BUFF地址(对象的句柄)
  DWORD dwMilliseconds       //最长等候时间,当设置为INIFINITE时,永无超时
);

MSDN官方文档解释WaitForSingleObject函数

当无信号时,该函数为阻塞函数。

WaitForMultipleObjects(
  DWORD nCount,               //句柄数量
  CONST HANDLE *lpHandles,    //句柄数组地址
  BOOL bWaitAll,              //等候方式
  DWORD dwMilliseconds        //最大等候时间(如果设定为INFINITE,则永无超时
);
bWaitAll--等候方式:
 TRUE:所有句柄有信号才结束等候
 FALSE:所有句柄中只要有一个有信号,就结束等候


相关文章
|
5月前
|
安全 Java 编译器
深入理解PHP 8.0新特性及其对开发的影响Java中的多线程编程:从理论到实践
【5月更文挑战第27天】在这篇文章中,我们将详细探讨PHP 8.0的新特性以及它们如何影响开发者的工作流程。我们将深入研究这些新特性,包括JIT编译器,联合类型,命名参数,以及更多的错误处理机制。我们还将讨论这些新特性如何提高代码的可读性和性能,以及它们如何改变我们编写和维护PHP应用程序的方式。 【5月更文挑战第27天】在现代计算机科学中,多线程编程是一个重要的概念,它允许多个线程在同一时间内运行,从而提高了程序的效率和性能。本文将深入探讨Java中的多线程编程,包括其理论基础,实现方法,以及一些常见的问题和解决方案。我们将通过实例来理解如何创建和管理线程,以及如何使用Java的并发工具来
|
19天前
|
API Android开发 iOS开发
安卓与iOS开发中的线程管理对比
【9月更文挑战第12天】在移动应用的世界中,安卓和iOS平台各自拥有庞大的用户群体。开发者们在这两个平台上构建应用时,线程管理是他们必须面对的关键挑战之一。本文将深入探讨两大平台在线程管理方面的异同,通过直观的代码示例,揭示它们各自的设计理念和实现方式,帮助读者更好地理解如何在安卓与iOS开发中高效地处理多线程任务。
|
4天前
|
存储 安全 程序员
Windows任务管理器开发原理与实现
Windows任务管理器开发原理与实现
|
2月前
|
Java Windows
【Azure Developer】Windows中通过pslist命令查看到Java进程和线程信息,但为什么和代码中打印出来的进程号不一致呢?
【Azure Developer】Windows中通过pslist命令查看到Java进程和线程信息,但为什么和代码中打印出来的进程号不一致呢?
|
2月前
|
安全 测试技术 调度
iOS开发-多线程编程
【8月更文挑战第12天】在iOS开发中,属性的内存管理至关重要,直接影响应用性能与稳定性。主要策略包括:`strong`(强引用),保持对象不被释放;`weak`(弱引用),不保持对象,有助于避免循环引用;`assign`(赋值),适用于基本数据类型及非指针对象类型;`copy`(复制),复制对象而非引用,确保不变性。内存管理基于引用计数,利用自动引用计数(ARC)自动管理对象生命周期。此外,需注意避免循环引用,特别是在block中。最佳实践包括理解各策略、避免不必要的强引用、及时释放不再使用的对象、注意block中的内存管理,并使用工具进行内存分析。正确管理内存能显著提升应用质量。
|
2月前
|
开发者 C# Windows
WPF与游戏开发:当桌面应用遇见游戏梦想——利用Windows Presentation Foundation打造属于你的2D游戏世界,从环境搭建到代码实践全面解析新兴开发路径
【8月更文挑战第31天】随着游戏开发技术的进步,WPF作为.NET Framework的一部分,凭借其图形渲染能力和灵活的UI设计,成为桌面游戏开发的新选择。本文通过技术综述和示例代码,介绍如何利用WPF进行游戏开发。首先确保安装最新版Visual Studio并创建WPF项目。接着,通过XAML设计游戏界面,并在C#中实现游戏逻辑,如玩家控制和障碍物碰撞检测。示例展示了创建基本2D游戏的过程,包括角色移动和碰撞处理。通过本文,WPF开发者可更好地理解并应用游戏开发技术,创造吸引人的桌面游戏。
88 0
|
2月前
|
开发者 iOS开发 C#
Uno Platform 入门超详细指南:从零开始教你打造兼容 Web、Windows、iOS 和 Android 的跨平台应用,轻松掌握 XAML 与 C# 开发技巧,快速上手示例代码助你迈出第一步
【8月更文挑战第31天】Uno Platform 是一个基于 Microsoft .NET 的开源框架,支持使用 C# 和 XAML 构建跨平台应用,适用于 Web(WebAssembly)、Windows、Linux、macOS、iOS 和 Android。它允许开发者共享几乎全部的业务逻辑和 UI 代码,同时保持原生性能。选择 Uno Platform 可以统一开发体验,减少代码重复,降低开发成本。安装时需先配置好 Visual Studio 或 Visual Studio for Mac,并通过 NuGet 或官网下载工具包。
47 0
|
2月前
|
安全 网络安全 数据安全/隐私保护
网络安全与信息安全:关于网络安全漏洞、加密技术、安全意识等方面的知识分享安卓与iOS开发中的线程管理比较
【8月更文挑战第30天】本文将探讨网络安全与信息安全的重要性,并分享关于网络安全漏洞、加密技术和安全意识的知识。我们将了解常见的网络攻击类型和防御策略,以及如何通过加密技术和提高安全意识来保护个人和组织的信息安全。
|
3月前
|
Linux Apache C++
FFmpeg开发笔记(三十五)Windows环境给FFmpeg集成libsrt
该文介绍了如何在Windows环境下为FFmpeg集成SRT协议支持库libsrt。首先,需要安装Perl和Nasm,然后编译OpenSSL。接着,下载libsrt源码并使用CMake配置,生成VS工程并编译生成srt.dll和srt.lib。最后,将编译出的库文件和头文件按照特定目录结构放置,并更新环境变量,重新配置启用libsrt的FFmpeg并进行编译安装。该过程有助于优化直播推流的性能,减少卡顿问题。
86 2
FFmpeg开发笔记(三十五)Windows环境给FFmpeg集成libsrt
|
3月前
|
缓存 编译器 Go
开发与运维线程问题之Go语言的goroutine基于线程模型实现如何解决
开发与运维线程问题之Go语言的goroutine基于线程模型实现如何解决
51 3

相关课程

更多