Linux和windows进程同步与线程同步那些事儿(四):windows 下进程同步

2024-01-11 19:52

本文主要是介绍Linux和windows进程同步与线程同步那些事儿(四):windows 下进程同步,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Linux和windows进程同步与线程同步那些事儿(一)
Linux和windows进程同步与线程同步那些事儿(二): windows线程同步详解示例
Linux和windows进程同步与线程同步那些事儿(三): Linux线程同步详解示例
Linux和windows进程同步与线程同步那些事儿(四):windows 下进程同步
Linux和windows进程同步与线程同步那些事儿(五):Linux下进程同步


1. 互斥量(Mutex):

互斥量是最常用的线程同步机制,它可以确保在同一时间只有一个线程可以访问共享资源。
在Windows中,可以使用CreateMutex函数来创建互斥量。

下面是一个使用互斥量实现进程同步的代码示例。该示例展示了两个进程之间如何使用互斥量来保证共享资源的互斥访问。

#include <iostream>
#include <windows.h>int main() {// 创建互斥量对象HANDLE hMutex = CreateMutex(NULL, FALSE, "MyMutex");if (hMutex == NULL) {std::cout << "互斥量创建失败!" << std::endl;return 1;}// 尝试获取互斥量的所有权DWORD dwWaitResult = WaitForSingleObject(hMutex, INFINITE);if (dwWaitResult == WAIT_OBJECT_0) {std::cout << "进程A获得了互斥量的所有权!" << std::endl;// 模拟共享资源的访问std::cout << "进程A正在访问共享资源..." << std::endl;Sleep(5000);// 释放互斥量的所有权ReleaseMutex(hMutex);std::cout << "进程A释放了互斥量的所有权!" << std::endl;}else {std::cout << "进程A无法获取互斥量的所有权!" << std::endl;}// 关闭互斥量句柄CloseHandle(hMutex);return 0;
}
#include <iostream>
#include <windows.h>int main() {// 创建互斥量对象HANDLE hMutex = CreateMutex(NULL, FALSE, "MyMutex");if (hMutex == NULL) {std::cout << "互斥量创建失败!" << std::endl;return 1;}// 尝试获取互斥量的所有权DWORD dwWaitResult = WaitForSingleObject(hMutex, INFINITE);if (dwWaitResult == WAIT_OBJECT_0) {std::cout << "进程B获得了互斥量的所有权!" << std::endl;// 模拟共享资源的访问std::cout << "进程B正在访问共享资源..." << std::endl;Sleep(3000);// 释放互斥量的所有权ReleaseMutex(hMutex);std::cout << "进程B释放了互斥量的所有权!" << std::endl;}else {std::cout << "进程B无法获取互斥量的所有权!" << std::endl;}// 关闭互斥量句柄CloseHandle(hMutex);return 0;
}

在上面的代码示例中,进程A和进程B都通过调用CreateMutex函数来创建了一个名为"MyMutex"的互斥量对象。接着,它们分别调用WaitForSingleObject函数来尝试获取互斥量的所有权。

如果某个进程成功获取了互斥量的所有权(即dwWaitResult的值为WAIT_OBJECT_0),那么它就可以访问共享资源。在示例中,为了模拟共享资源的访问,我们使用了Sleep函数来暂停程序的执行。

完成对共享资源的访问后,进程需要调用ReleaseMutex函数来释放互斥量的所有权,以便其他进程可以获取互斥量的所有权。

最后,进程要调用CloseHandle函数来关闭互斥量句柄,释放系统资源。

请注意,这只是一个简单的示例,实际的应用中可能还需要处理异常情况、错误处理等。


2. 事件(Event):

事件用于线程间的通信和同步,允许线程等待某个特定事件的发生。
在Windows中,可以使用CreateEvent函数来创建事件对象。

下面是一个使用事件(Event)实现进程同步的代码示例。该示例展示了两个进程之间如何使用事件来进行进程同步。

#include <iostream>
#include <windows.h>int main() {// 创建事件对象HANDLE hEvent = CreateEvent(NULL, FALSE, FALSE, "MyEvent");if (hEvent == NULL) {std::cout << "事件创建失败!" << std::endl;return 1;}// 等待事件变为有信号状态DWORD dwWaitResult = WaitForSingleObject(hEvent, INFINITE);if (dwWaitResult == WAIT_OBJECT_0) {std::cout << "进程A检测到事件有信号!" << std::endl;// 执行需要同步的操作std::cout << "进程A正在执行同步操作..." << std::endl;Sleep(5000);std::cout << "进程A完成同步操作!" << std::endl;}else {std::cout << "进程A等待事件超时或发生其他错误!" << std::endl;}// 关闭事件句柄CloseHandle(hEvent);return 0;
}
#include <iostream>
#include <windows.h>int main() {// 打开事件对象HANDLE hEvent = OpenEvent(EVENT_MODIFY_STATE, FALSE, "MyEvent");if (hEvent == NULL) {std::cout << "事件打开失败!" << std::endl;return 1;}// 设置事件为有信号状态if (SetEvent(hEvent)) {std::cout << "进程B设置了事件的信号!" << std::endl;}else {std::cout << "进程B设置事件的信号失败!" << std::endl;}// 关闭事件句柄CloseHandle(hEvent);return 0;
}

在上面的代码示例中,进程A通过调用CreateEvent函数来创建了一个名为"MyEvent"的事件对象。而进程B则通过调用OpenEvent函数来打开同一个事件对象。

进程A调用WaitForSingleObject函数来等待事件变为有信号状态。一旦事件有信号,进程A就可以执行需要同步的操作。在示例中,我们使用了Sleep函数来模拟同步操作的执行。

进程B在需要进行同步的时候,调用SetEvent函数来设置事件的信号。这样,进程A就会被唤醒,继续执行同步操作。

请注意,事件对象具有自动复位(auto-reset)的特性,即一旦某个等待事件的进程被唤醒,事件会自动重置为无信号状态,其他等待事件的进程将继续等待。

最后,进程要调用CloseHandle函数来关闭事件句柄,释放系统资源。


3. 临界区(Critical Section):

临界区用于保护共享资源,确保在同一时间只有一个线程可以访问。
在Windows中,可以使用InitializeCriticalSection函数来初始化临界区。

下面是一个使用临界区(Critical Section)实现进程同步的代码示例。该示例展示了两个进程之间如何使用临界区来进行进程同步。

#include <iostream>
#include <windows.h>CRITICAL_SECTION g_cs;  // 全局临界区对象int main() {// 初始化临界区InitializeCriticalSection(&g_cs);// 进入临界区EnterCriticalSection(&g_cs);std::cout << "进程A进入临界区,执行同步操作..." << std::endl;Sleep(5000);  // 模拟同步操作std::cout << "进程A完成同步操作,退出临界区!" << std::endl;// 离开临界区LeaveCriticalSection(&g_cs);// 销毁临界区DeleteCriticalSection(&g_cs);return 0;
}
#include <iostream>
#include <windows.h>CRITICAL_SECTION g_cs;  // 全局临界区对象int main() {// 初始化临界区InitializeCriticalSection(&g_cs);// 进入临界区EnterCriticalSection(&g_cs);std::cout << "进程B进入临界区,执行同步操作..." << std::endl;Sleep(5000);  // 模拟同步操作std::cout << "进程B完成同步操作,退出临界区!" << std::endl;// 离开临界区LeaveCriticalSection(&g_cs);// 销毁临界区DeleteCriticalSection(&g_cs);return 0;
}

在上面的代码示例中,我们使用CRITICAL_SECTION类型的全局变量g_cs作为临界区对象。

进程A和进程B都会初始化临界区对象,进入临界区进行同步操作。

在示例中,我们使用了Sleep函数来模拟同步操作的执行。

进程A和进程B在完成同步操作后,都需要离开临界区。

最后,进程要调用DeleteCriticalSection函数来销毁临界区对象,释放系统资源。

请注意,临界区是进程内的同步机制,不适用于跨进程的同步。如果需要跨进程的同步,可以考虑使用互斥量(Mutex)或事件(Event)等其他机制。


4. 信号量(Semaphore):

信号量是一种经典的线程同步机制,它可以用于控制对共享资源的访问。
在Windows中,可以使用CreateSemaphore函数来创建信号量。

下面是一个使用信号量(Semaphore)实现进程同步的代码示例。该示例展示了两个进程之间如何使用信号量来进行进程同步。

#include <iostream>
#include <windows.h>HANDLE g_semaphore;  // 全局信号量句柄int main() {// 创建一个信号量,初始值为0,用于进程同步g_semaphore = CreateSemaphore(NULL, 0, 1, L"ProcessSync");std::cout << "进程A执行同步操作..." << std::endl;Sleep(5000);  // 模拟同步操作std::cout << "进程A完成同步操作,释放信号量!" << std::endl;ReleaseSemaphore(g_semaphore, 1, NULL);  // 释放信号量,允许进程B执行// 关闭信号量句柄CloseHandle(g_semaphore);return 0;
}
#include <iostream>
#include <windows.h>HANDLE g_semaphore;  // 全局信号量句柄int main() {// 创建一个信号量,初始值为0,用于进程同步g_semaphore = CreateSemaphore(NULL, 0, 1, L"ProcessSync");// 进程B等待信号量,直到进程A释放信号量WaitForSingleObject(g_semaphore, INFINITE);std::cout << "进程B执行同步操作..." << std::endl;Sleep(5000);  // 模拟同步操作std::cout << "进程B完成同步操作,释放信号量!" << std::endl;ReleaseSemaphore(g_semaphore, 1, NULL);  // 释放信号量,允许其他进程执行// 关闭信号量句柄CloseHandle(g_semaphore);return 0;
}

在上面的示例代码中,我们使用了HANDLE类型的全局变量g_semaphore作为信号量句柄。
进程A创建了一个名为"ProcessSync"的信号量,并执行同步操作之后,调用ReleaseSemaphore函数释放信号量。

进程B在等待信号量状态下,调用WaitForSingleObject函数等待进程A释放信号量,然后执行同步操作。
最后,进程要调用CloseHandle函数来关闭信号量句柄,释放系统资源。

请注意,信号量是一种用于进程间同步的机制,可以在不同进程之间实现同步。在示例中,我们使用了一个简单的二进制信号量,它的初始值为0,进程A通过ReleaseSemaphore函数将其值设置为1,使进程B可以继续执行。


5. 命名管道(Named Pipe):

命名管道是一种进程间通信的机制,可以用于实现进程间的数据交换和同步。
在Windows中,可以使用CreateNamedPipe函数来创建命名管道。

在 Windows 下,可以使用命名管道(Named Pipe)实现多进程间的同步。命名管道是一种特殊的文件,用于在不同进程间进行通信。下面是一个使用命名管道实现多进程同步的 C++ 代码示例:

进程A:写入数据

#include <iostream>
#include <windows.h>int main()
{HANDLE hPipe;// 创建命名管道LPCWSTR pipeName = TEXT("\\\\.\\pipe\\my_pipe");hPipe = CreateNamedPipe(pipeName,PIPE_ACCESS_DUPLEX,PIPE_TYPE_MESSAGE | PIPE_WAIT,1,              // 最大实例数65536,          // 输出缓冲区大小65536,          // 输入缓冲区大小0,              // 默认超时时间NULL            // 安全属性);// 等待进程B连接ConnectNamedPipe(hPipe, NULL);// 向管道写入数据const char* data = "Hello, World!";DWORD bytesWritten;WriteFile(hPipe, data, strlen(data) + 1, &bytesWritten, NULL);// 关闭管道DisconnectNamedPipe(hPipe);CloseHandle(hPipe);return 0;
}

进程B:读取数据

#include <iostream>
#include <windows.h>int main()
{HANDLE hPipe;// 连接到命名管道LPCWSTR pipeName = TEXT("\\\\.\\pipe\\my_pipe");hPipe = CreateFile(pipeName,GENERIC_READ,0,              // 共享模式NULL,           // 安全属性OPEN_EXISTING,0,              // 属性和标志NULL            // 模板文件);// 从管道读取数据char buffer[65536];DWORD bytesRead;ReadFile(hPipe, buffer, sizeof(buffer), &bytesRead, NULL);// 打印数据std::cout << buffer << std::endl;// 关闭管道CloseHandle(hPipe);return 0;
}

上述代码中,进程A创建了一个命名管道,并等待进程B连接。然后,它向管道写入数据。进程B连接到同一个命名管道,并读取进程A写入的数据。

让我们更详细地解释一下代码:

  1. 在进程A中,CreateNamedPipe 函数用于创建命名管道。参数 pipeName 指定了命名管道的名称,格式为 \\.\pipe\pipeName。其他参数设置了管道的访问权限、缓冲区大小等。

  2. 进程A调用 ConnectNamedPipe 函数等待进程B连接。

  3. 进程A使用 WriteFile 函数向管道写入数据。

  4. 进程B使用 CreateFile 函数连接到命名管道。

  5. 进程B调用 ReadFile 函数从管道中读取数据。

  6. 最后,进程B关闭了管道。

在实际使用中,可以根据需要对代码进行修改和扩展,以满足你的具体需求。例如,你可以使用多线程创建多个进程,或者通过命名管道进行双向通信。


6. 共享内存(Shared Memory):

共享内存允许多个进程访问同一块内存区域,从而实现进程间的数据共享和通信。
在Windows中,可以使用CreateFileMappingMapViewOfFile等函数来创建和访问共享内存区域。

下面是一个使用共享内存在Windows下实现多个进程同步的代码示例:

#include <iostream>
#include <Windows.h>// 共享内存结构体
struct SharedData {HANDLE mutex;         // 互斥量HANDLE event;         // 事件CONDITION_VARIABLE conditionVariable; // 条件变量int counter;          // 共享计数器
};int main() {// 创建共享内存HANDLE hMapFile = CreateFileMapping(INVALID_HANDLE_VALUE,nullptr,PAGE_READWRITE,0,sizeof(SharedData),L"MySharedMemory");if (hMapFile == NULL) {std::cerr << "Failed to create shared memory" << std::endl;return 1;}// 获取共享内存指针SharedData* sharedData = (SharedData*)MapViewOfFile(hMapFile,FILE_MAP_ALL_ACCESS,0,0,sizeof(SharedData));if (sharedData == NULL) {std::cerr << "Failed to map shared memory" << std::endl;CloseHandle(hMapFile);return 1;}// 初始化互斥量、事件和条件变量sharedData->mutex = CreateMutex(nullptr, FALSE, nullptr);sharedData->event = CreateEvent(nullptr, FALSE, FALSE, nullptr);InitializeConditionVariable(&sharedData->conditionVariable);sharedData->counter = 0;// 创建多个进程进行同步操作for (int i = 0; i < 5; i++) {STARTUPINFO si = {};PROCESS_INFORMATION pi = {};wchar_t cmdLine[] = L"ChildProcess.exe"; // 子进程程序名// 启动子进程if (!CreateProcess(nullptr,cmdLine,nullptr,nullptr,FALSE,0,nullptr,nullptr,&si,&pi)) {std::cerr << "Failed to create child process" << std::endl;CloseHandle(hMapFile);return 1;}// 关闭子进程句柄CloseHandle(pi.hThread);CloseHandle(pi.hProcess);}// 等待所有子进程执行完毕WaitForMultipleObjects(5, sharedData->processHandles, TRUE, INFINITE);// 输出计数器值std::cout << "Counter = " << sharedData->counter << std::endl;// 释放资源UnmapViewOfFile(sharedData);CloseHandle(sharedData->mutex);CloseHandle(sharedData->event);CloseHandle(hMapFile);return 0;
}

这个示例中,首先创建了一个共享内存来存储互斥量、事件和条件变量等同步对象以及一个共享计数器。然后创建了多个子进程,在子进程中进行同步操作。

子进程代码示例(ChildProcess.exe):

#include <iostream>
#include <Windows.h>struct SharedData {HANDLE mutex;HANDLE event;CONDITION_VARIABLE conditionVariable;int counter;
};int main() {HANDLE hMapFile = OpenFileMapping(FILE_MAP_ALL_ACCESS,FALSE,L"MySharedMemory");if (hMapFile == NULL) {std::cerr << "Failed to open shared memory" << std::endl;return 1;}SharedData* sharedData = (SharedData*)MapViewOfFile(hMapFile,FILE_MAP_ALL_ACCESS,0,0,sizeof(SharedData));if (sharedData == NULL) {std::cerr << "Failed to map shared memory" << std::endl;CloseHandle(hMapFile);return 1;}// 等待主进程通知WaitForSingleObject(sharedData->event, INFINITE);// 互斥量保护临界区WaitForSingleObject(sharedData->mutex, INFINITE);// 执行同步操作sharedData->counter++;std::cout << "Child process counter = " << sharedData->counter << std::endl;// 离开临界区ReleaseMutex(sharedData->mutex);// 通知主进程完成WakeConditionVariable(&sharedData->conditionVariable);// 释放资源UnmapViewOfFile(sharedData);CloseHandle(hMapFile);return 0;
}

在子进程中,首先打开共享内存,并获取共享内存指针。然后等待主进程通知,获取互斥量保护临界区,执行同步操作,离开临界区,通知主进程完成,并释放资源。


总结

互斥量(mutex):
适用场景:多个进程需要争夺同一个资源时,只能有一个进程访问资源。
优点:简单易用,能够实现多个进程间的互斥访问。
缺点:性能较低,需要进行系统调用,可能引起上下文切换开销。

事件(event):
适用场景:多个进程需要等待某个特定事件的发生后才能继续执行。
优点:可以在多个进程之间进行高效的同步。
缺点:只能用于单个事件的同步,不能用于争夺资源的同步。

临界区(critical section):
适用场景:多个进程需要访问共享资源时,只能有一个进程访问资源。
优点:具有较高的性能,适用于保护共享资源的场景。
缺点:只能用于同一个进程内的进程同步,不能用于跨进程同步。

信号量(semaphore):
适用场景:多个进程需要访问有限数量的资源时,控制同时访问的进程数量。
优点:可以用于控制同时访问资源的进程数量。
缺点:较复杂,容易引发死锁问题。

命名管道(named pipe):
适用场景:多个进程之间进行双向通信和同步。
优点:提供了进程间的可靠通信和同步机制。
缺点:性能较低,只能用于进程之间的通信和同步。

共享内存(shared memory):
适用场景:多个进程需要共享大块数据时。
优点:高效的进程间通信,能够实现数据共享。
缺点:需要进行同步,容易引发数据不一致或竞态条件。

综合比较:

  • 互斥量 : 适用于争夺单个资源的情况,使用简单,但性能较低。
  • 事件 : 适用于等待特定事件发生后才能继续执行的场景,同步效率高。
  • 临界区: 适用于保护共享资源的场景,性能较高。
  • 信号量: 适用于控制同时访问资源的进程数量,但使用较复杂,容易产生死锁。
  • 命名管道: 适用于进程间的通信和同步,但性能较低。
  • 共享内存: 适用于共享大块数据的场景,需要进行同步操作,可能引发竞态条件。

根据具体的场景和需求,可以选择合适的进程同步机制来实现进程间的同步和通信。

这篇关于Linux和windows进程同步与线程同步那些事儿(四):windows 下进程同步的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/595523

相关文章

linux生产者,消费者问题

pthread_cond_wait() :用于阻塞当前线程,等待别的线程使用pthread_cond_signal()或pthread_cond_broadcast来唤醒它。 pthread_cond_wait() 必须与pthread_mutex 配套使用。pthread_cond_wait()函数一进入wait状态就会自动release mutex。当其他线程通过pthread

Linux 安装、配置Tomcat 的HTTPS

Linux 安装 、配置Tomcat的HTTPS 安装Tomcat 这里选择的是 tomcat 10.X ,需要Java 11及更高版本 Binary Distributions ->Core->选择 tar.gz包 下载、上传到内网服务器 /opt 目录tar -xzf 解压将解压的根目录改名为 tomat-10 并移动到 /opt 下, 形成个人习惯的路径 /opt/tomcat-10

RedHat运维-Linux文本操作基础-AWK进阶

你不用整理,跟着敲一遍,有个印象,然后把它保存到本地,以后要用再去看,如果有了新东西,你自个再添加。这是我参考牛客上的shell编程专项题,只不过换成了问答的方式而已。不用背,就算是我自己亲自敲,我现在好多也记不住。 1. 输出nowcoder.txt文件第5行的内容 2. 输出nowcoder.txt文件第6行的内容 3. 输出nowcoder.txt文件第7行的内容 4. 输出nowcode

【Linux进阶】UNIX体系结构分解——操作系统,内核,shell

1.什么是操作系统? 从严格意义上说,可将操作系统定义为一种软件,它控制计算机硬件资源,提供程序运行环境。我们通常将这种软件称为内核(kerel),因为它相对较小,而且位于环境的核心。  从广义上说,操作系统包括了内核和一些其他软件,这些软件使得计算机能够发挥作用,并使计算机具有自己的特生。这里所说的其他软件包括系统实用程序(system utility)、应用程序、shell以及公用函数库等

问题-windows-VPN不正确关闭导致网页打不开

为什么会发生这类事情呢? 主要原因是关机之前vpn没有关掉导致的。 至于为什么没关掉vpn会导致网页打不开,我猜测是因为vpn建立的链接没被更改。 正确关掉vpn的时候,会把ip链接断掉,如果你不正确关掉,ip链接没有断掉,此时你vpn又是没启动的,没有域名解析,所以就打不开网站。 你可以在打不开网页的时候,把vpn打开,你会发现网络又可以登录了。 方法一 注意:方法一虽然方便,但是可能会有

Windows/macOS/Linux 安装 Redis 和 Redis Desktop Manager 可视化工具

本文所有安装都在macOS High Sierra 10.13.4进行,Windows安装相对容易些,Linux安装与macOS类似,文中会做区分讲解 1. Redis安装 1.下载Redis https://redis.io/download 把下载的源码更名为redis-4.0.9-source,我喜欢跟maven、Tomcat放在一起,就放到/Users/zhan/Documents

Linux系统稳定性的奥秘:探究其背后的机制与哲学

在计算机操作系统的世界里,Linux以其卓越的稳定性和可靠性著称,成为服务器、嵌入式系统乃至个人电脑用户的首选。那么,是什么造就了Linux如此之高的稳定性呢?本文将深入解析Linux系统稳定性的几个关键因素,揭示其背后的技术哲学与实践。 1. 开源协作的力量Linux是一个开源项目,意味着任何人都可以查看、修改和贡献其源代码。这种开放性吸引了全球成千上万的开发者参与到内核的维护与优化中,形成了

Windows中,.net framework 3.5安装

安装.net framework,目前已知2种方法,如下: 一、在MSDN下载对应的安装包,安装,这种可能无法安装成功,概率很大,不成功使用第二种方法,基本上没问题。 二、win8/8.1/10 下安装 .net framework 3.5.1: 1. 打开 win8/8.1/10 安装盘(这里指系统安装镜像文件),提取 sources\sxs 文件夹到 X:\sources\sxs (X代

时间服务器中,适用于国内的 NTP 服务器地址,可用于时间同步或 Android 加速 GPS 定位

NTP 是什么?   NTP 是网络时间协议(Network Time Protocol),它用来同步网络设备【如计算机、手机】的时间的协议。 NTP 实现什么目的?   目的很简单,就是为了提供准确时间。因为我们的手表、设备等,经常会时间跑着跑着就有误差,或快或慢的少几秒,时间长了甚至误差过分钟。 NTP 服务器列表 最常见、熟知的就是 www.pool.ntp.org/zo

Windows 可变刷新率是什么?如何开启?

在现代计算设备中,显示屏的刷新率对用户体验起着至关重要的作用。随着显示技术的不断进步,固定刷新率显示器逐渐被支持可变刷新率(Variable Refresh Rate, VRR)技术的显示器所取代。 可变刷新率定义 可变刷新率是什么?可变刷新率(VRR)是一种显示技术,它允许显示器的刷新率动态调整,以匹配显卡输出的帧率。传统的显示器通常具有固定的刷新率(如60Hz、75Hz等),这意味着显示器