首页 > 系统相关 >Windows多线程编程 互斥量和临界区使用

Windows多线程编程 互斥量和临界区使用

时间:2024-10-11 17:20:20浏览次数:11  
标签:NULL Windows void include XThread 互斥 线程 ThreadFunc 多线程

Windows 多线程编程允许程序同时运行多个线程,提高程序的并发性和执行效率。多线程编程中的核心概念包括线程的创建、同步、调度、数据共享和竞争条件等。本文详细介绍了 Windows 多线程编程的关键技术点,并解释如何使用线程同步机制来保证线程安全。

1. 线程基础概念

1.1 线程

线程是操作系统能够独立调度的最小执行单元。一个进程可以包含多个线程,这些线程共享进程的地址空间和资源。多线程编程通过并发执行多个线程,提升程序性能,特别是在 I/O 操作、网络请求或图像处理等任务中。

1.2 进程 vs. 线程

进程:程序在操作系统中的运行实例。每个进程有独立的地址空间和资源。
线程:线程是进程中的轻量级执行单元,多个线程可以共享进程的内存和资源。一个进程至少包含一个主线程,可以派生出多个子线程。

2. 线程的创建与管理

在 Windows 中,创建和管理线程可以通过 WinAPI 提供的多种方法,其中常用的是 CreateThread 和 C++11 提供的标准库线程类。

2.1 使用 CreateThread

这是 WinAPI 中直接用于创建线程的函数,它返回一个线程句柄,用于管理线程。

#include <windows.h>
#include <iostream>

// 线程函数
DWORD WINAPI ThreadFunc(LPVOID lpParam) {
    for (int i = 0; i < 5; i++) {
        std::cout << "Thread running...\n";
        Sleep(1000); // 模拟工作,暂停1秒
    }
    return 0;
}

int main() {
    HANDLE hThread = CreateThread(
        NULL,            // 默认安全属性
        0,               // 默认堆栈大小
        ThreadFunc,      // 线程函数
        NULL,            // 参数传递给线程函数
        0,               // 默认创建标志
        NULL             // 可选的线程ID
    );

    if (hThread == NULL) {
        std::cout << "Error: Unable to create thread\n";
        return 1;
    }

    // 等待线程结束
    WaitForSingleObject(hThread, INFINITE);

    // 关闭线程句柄
    CloseHandle(hThread);

    return 0;
}

2.2 使用 C++11 std::thread

现代 C++ 提供了跨平台的 std::thread 类,用来简化线程的创建和管理。

#include <iostream>
#include <thread>

void ThreadFunc() {
    for (int i = 0; i < 5; i++) {
        std::cout << "Thread running...\n";
        std::this_thread::sleep_for(std::chrono::seconds(1));  // 模拟工作
    }
}

int main() {
    std::thread t(ThreadFunc);  // 创建线程
    t.join();  // 等待线程结束

    return 0;
}

3. 线程同步

在多线程程序中,多个线程可能会同时访问共享资源(如内存、文件等),如果不加以控制,可能会导致数据竞态条件(Race Condition)。线程同步用于协调线程对共享资源的访问,避免数据冲突。

  • 常用的同步机制包括:
临界区(Critical Section)
互斥量(Mutex)
信号量(Semaphore)
事件(Event)

3.1 临界区(Critical Section)

临界区是一种轻量级的同步机制,仅适用于单进程的线程同步。临界区在同一时间只允许一个线程进入,其他线程必须等待当前线程离开临界区后才能进入。

#include <windows.h>
#include <iostream>

CRITICAL_SECTION criticalSection;  // 定义临界区

void ThreadFunc() {
    EnterCriticalSection(&criticalSection);  // 进入临界区
    std::cout << "Thread ID: " << GetCurrentThreadId() << " is working.\n";
    LeaveCriticalSection(&criticalSection);  // 离开临界区
}

int main() {
    InitializeCriticalSection(&criticalSection);  // 初始化临界区

    HANDLE hThread1 = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)ThreadFunc, NULL, 0, NULL);
    HANDLE hThread2 = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)ThreadFunc, NULL, 0, NULL);

    WaitForSingleObject(hThread1, INFINITE);
    WaitForSingleObject(hThread2, INFINITE);

    DeleteCriticalSection(&criticalSection);  // 删除临界区
    return 0;
}

3.2 互斥量(Mutex)

互斥量可以在多个线程甚至多个进程之间同步访问共享资源。与临界区相比,互斥量开销较大,但功能更强。

#include <windows.h>
#include <iostream>

HANDLE hMutex;

void ThreadFunc() {
    WaitForSingleObject(hMutex, INFINITE);  // 获取互斥量
    std::cout << "Thread ID: " << GetCurrentThreadId() << " is working.\n";
    ReleaseMutex(hMutex);  // 释放互斥量
}

int main() {
    hMutex = CreateMutex(NULL, FALSE, NULL);  // 创建互斥量

    HANDLE hThread1 = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)ThreadFunc, NULL, 0, NULL);
    HANDLE hThread2 = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)ThreadFunc, NULL, 0, NULL);

    WaitForSingleObject(hThread1, INFINITE);
    WaitForSingleObject(hThread2, INFINITE);

    CloseHandle(hMutex);  // 关闭互斥量句柄
    return 0;
}

3.3 信号量(Semaphore)

信号量允许多个线程访问同一资源,信号量内部有一个计数器,控制同时访问的线程数量。当计数器减为 0 时,其他线程必须等待。

#include <windows.h>
#include <iostream>

HANDLE hSemaphore;

void ThreadFunc() {
    WaitForSingleObject(hSemaphore, INFINITE);  // 等待信号量
    std::cout << "Thread ID: " << GetCurrentThreadId() << " is working.\n";
    ReleaseSemaphore(hSemaphore, 1, NULL);  // 释放信号量
}

int main() {
    hSemaphore = CreateSemaphore(NULL, 2, 2, NULL);  // 最大允许2个线程同时执行

    HANDLE hThread1 = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)ThreadFunc, NULL, 0, NULL);
    HANDLE hThread2 = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)ThreadFunc, NULL, 0, NULL);
    HANDLE hThread3 = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)ThreadFunc, NULL, 0, NULL);

    WaitForSingleObject(hThread1, INFINITE);
    WaitForSingleObject(hThread2, INFINITE);
    WaitForSingleObject(hThread3, INFINITE);

    CloseHandle(hSemaphore);
    return 0;
}

3.4 事件(Event)

事件用于在线程之间传递信号,某个线程可以等待事件的状态(有信号或无信号),然后作出相应动作。事件可以用来实现线程之间的通知机制。

#include <windows.h>
#include <iostream>

HANDLE hEvent;

DWORD WINAPI ThreadFunc(LPVOID lpParam) {
    std::cout << "Thread waiting for event...\n";
    WaitForSingleObject(hEvent, INFINITE);  // 等待事件
    std::cout << "Thread received event signal!\n";
    return 0;
}

int main() {
    hEvent = CreateEvent(NULL, TRUE, FALSE, NULL);  // 创建事件

    HANDLE hThread = CreateThread(NULL, 0, ThreadFunc, NULL, 0, NULL);

    Sleep(2000);  // 模拟工作
    SetEvent(hEvent);  // 触发事件

    WaitForSingleObject(hThread, INFINITE);
    CloseHandle(hEvent);
    return 0;
}

4. 线程同步中的问题

4.1 死锁

死锁是指两个或多个线程在等待彼此持有的资源,导致线程永远无法继续执行。避免死锁的方法:

使用一致的锁顺序:所有线程获取锁的顺序要一致。
避免嵌套锁定:尽量避免一个线程在持有一个锁的同时请求另一个锁。

4.2 竞争条件

竞争条件发生在多个线程同时读取或写入共享数据时,由于执行顺序的不确定性,可能导致错误结果。解决方法是使用合适的同步机制(如临界区、互斥量等)来保护共享数据的访问。

5、Window线程和临界封装

1、线程XThread

1、XThread.h
#pragma once

#ifdef  XPLATFORM_EXPORTS
#define XPLATFORM_API __declspec(dllexport)
#else
#define XPLATFORM_API __declspec(dllimport)
#endif


class XPLATFORM_API XThread
{
public:
	XThread();
	virtual ~XThread();
	bool Start();
	virtual void Run() = 0;
	void Wait();
	void Suspend();
	void Resume();
private:
	unsigned int thId = 0;
};


2、XThread.cpp
#include "XThread.h"
#include <process.h>
#include <Windows.h>

XThread::XThread()
{

}
XThread::~XThread()
{
}
static void ThreadMain(void *para)
{
	XThread* th = (XThread*)para;
	if(th == nullptr) return;
	th->Run();
	_endthread();
}
bool XThread::Start()
{
	thId = _beginthread(ThreadMain, 0, this);
	return thId <= 0;
}
void XThread::Wait()
{
	if(thId <= 0) return;
	WaitForSingleObject((HANDLE)thId, INFINITE);
}
void XThread::Suspend()
{
	if (thId <= 0) return;
	SuspendThread((HANDLE)thId);
}
void XThread::Resume()
{
	if (thId <= 0) return;
	ResumeThread((HANDLE)thId);
}

2、临界区封装

1、XMutex.h
#pragma once
#ifdef  XPLATFORM_EXPORTS
#define XPLATFORM_API __declspec(dllexport)
#else
#define XPLATFORM_API __declspec(dllimport)
#endif
class XPLATFORM_API XMutex
{
public:
	XMutex();
	~XMutex();
	void Lock();
	void UnLock();
private:
	void* section = nullptr;
};
2、XMutex.cpp
#include "XMutex.h"
#include <windows.h>

XMutex::XMutex()
{
	this->section = new CRITICAL_SECTION();
	if (section == nullptr) return;
	InitializeCriticalSection((LPCRITICAL_SECTION)this->section);
}

XMutex::~XMutex()
{
	CRITICAL_SECTION *critical_section = (LPCRITICAL_SECTION)this->section;
	if(critical_section == nullptr) return;
	delete critical_section;
}

void XMutex::Lock()
{
	if (section == nullptr) return;
	EnterCriticalSection((LPCRITICAL_SECTION)this->section);
}

void XMutex::UnLock()
{
	if(section == nullptr) return;
	LeaveCriticalSection((LPCRITICAL_SECTION)this->section);
}

标签:NULL,Windows,void,include,XThread,互斥,线程,ThreadFunc,多线程
From: https://blog.csdn.net/weixin_42462436/article/details/142859151

相关文章

  • 钉钉内网穿透工具-免费windows版
    提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档前言当我们处异地时,如何高效地访问公司内部网络资源成为了一个挑战。钉钉作为企业通讯和协作平台,支持内网穿透!这意味着即使您身在外地,也能像坐在办公室一样访问公司的内部系统和数据。听起来是不是很方......
  • Windows11搭建Speedtest测速服务器
    在Windows11上配置Speedtest服务器下载本教程中所需要的软件列表开支在下载好以上软件后,下面开始正式进行服务器搭建所有软件打包地址1.在Windows11上安装ISS服务a.点击Start--->System--->Optionalfeature进入b.选择最下面的MoreWindowsfeaturec.勾选需要开......
  • 解决Windows触摸板频繁失灵问题(I2C HID无法启动 代码10)
    我的电脑是华硕的笔记本,在多年前之前就出现偶尔的触摸板失灵的情况,那时候在官网升级驱动啥的也能解决,但是不是长久之计,后来又是经常在网上看教程啥的,终于在今天找到了最好用的方法,没有之一,绝对是目前最有效,最好的解决方法,也是参考了别人的文章,结合自己的实际情况而总结。话不多......
  • [自用] 虚拟机windows11-x64,安装MySQL 8.0.32,记录
    前面忘截图了提示要求电脑里安装VS2015/2017/2019,但虚拟机里只有VS2013。网上说可以一起装,但是我虚拟机配置不太行,再说吧,不行用我自己笔记本,虽然也有点菜,但比虚拟机强。虚拟机配置安装之后的配置密码三个旧的特殊符号这少一步,写的是点击execute来应用配置apply......
  • SMB签名是一种通过数字签名技术保障数据在网络传输过程中的完整性和来源验证的机制。
    SMB签名是ServerMessageBlock(SMB)协议中的一种安全机制,旨在确保数据的完整性和身份验证。1.什么是SMB签名?SMB签名是一种通过数字签名技术保障数据在网络传输过程中的完整性和来源验证的机制。它通过对数据进行哈希处理,并附加一个签名,确保接收方能够确认收到的数据没有被篡改。......
  • Windows 或Office 激活失败 错误代码 0x80072F8F
    国庆节收假回来,工作站上几台电脑出现问题,重新修复,更换硬件(主板)或是重装系统,最终需要激活成为正版来使用。其中一台,在激活时,Windows激活失败错误代码0x80072F8F,发现日期与时间不正确: 把时间改正之后,终于激活Windows。 哈哈......在另外一台电脑上,Windows是激活的,但需要......
  • Windows-WMI 事件 ID 10或0x80041003——解决过程
    2024年10月8日国庆节后,第一天上班,实验室里一台PC机出现故障,Windows7系统,可以正常启动进入安全模式,但是正常启动无法加载桌面,可以看见鼠标,Ctrl+Alt+Del无法调出任务管理器。开始处理,进入安全模式,查看系统日志。发现一个错误如下(截取自[https://www.cnblogs.com/longware/p/78231......
  • kms激活Windows
    安装KMS服务个人是在软路由系统中安装的,安装请另外寻找服务器安装KMS教程使用命令行进行激活1.卸载当前激活的秘钥(管理员启动命令行)slmgr/upk2.安装新的秘钥秘钥在KMS软件的日志中,自行寻找slmgr/ipkXXXX-XXXX-XXXXXX3.设置KMS服务器地址可以输入内网地址slmg......
  • 又一款windows搜索神器!非常实用的文件搜索工具,可以说很智能了(带私活源码)
     哈喽,大家好,今天为大家介绍一款文件搜索神器!Listary简介Listary是一款非常实用的搜索工具,它可以为“我的电脑”(资源管理器)添加许多智能命令,包括收藏文件夹、快速打开最近浏览的文件夹、快速显示/隐藏文件扩展名等功能。这些实用功能可以帮助你在日常收藏和整理文件时提高效......
  • 全局视角看技术-Java多线程演进史
    作者:京东科技文涛全文较长共6468字,语言通俗易懂,是一篇具有大纲性质的关于多线程的梳理,作者从历史演进的角度讲了多线程相关知识体系,让你知其然知其所以然。前言2022年09月22日,JDK19发布了,此版本最大的亮点就是支持虚拟线程,从此轻量级线程家族再添一员大将。虚拟线程使JVM摆脱......