window下线程同步之(Critical Sections(关键代码段、关键区域、临界区域)

本文主要是介绍window下线程同步之(Critical Sections(关键代码段、关键区域、临界区域),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 关键区域(CriticalSection)

临界区是为了确保同一个代码片段在同一时间只能被一个线程访问,与原子锁不同的是临界区是多条指令的锁定,而原子锁仅仅对单条操作指令有效;临界区和原子锁只能控制同一个进程中线程的同步

使用方法:

1、初始化:InitializeCriticalSection; 
2、删除:DeleteCriticalSection; 
3、进入:EnterCriticalSection(可能造成阻塞); 
4、尝试进入:TryEnterCriticalSection(不会造成阻塞); 
5、离开:LeaveCriticalSection;

固有特点(优点+缺点):
1、是一个用户模式的对象,不是系统核心对象;
2、因为不是核心对象,所以执行速度快,有效率;
3、因为不是核心对象,所以不能跨进程使用;
4、可以多次“进入”,但必须多次“退出”;
5、最好不要同时进入或等待多个 Critical Sections,容易造成死锁;
6、无法检测到进入到 Critical Sections 里面的线程当前是否已经退出!

 

一般错误的情况:

#include <stdio.h> 
#include <windows.h>long g_nNum = 0 ; 
DWORD WINAPI ThreadProc(__in  LPVOID lpParameter); 
const int THREAD_NUM = 10;int main() 
{HANDLE  handle[THREAD_NUM];    g_nNum = 0; int var = 0; while ( var< THREAD_NUM) { handle[ var++] = CreateThread(NULL, 0, ThreadProc, NULL, 0, NULL);} WaitForMultipleObjects(THREAD_NUM, handle, TRUE, INFINITE);for( var=0; var<sizeof(handle); var++) { CloseHandle(handle[var]); }return 0; 
}DWORD WINAPI ThreadProc(__in  LPVOID lpParameter) 
{ Sleep(50); g_nNum++; Sleep(0); printf("当前计数为:%d\n",g_nNum); return 0; 
}

运行2次结果:

image

image

用了关键区域的情况:

#include <stdio.h> 
#include <windows.h>long g_nNum = 0 ; 
DWORD WINAPI ThreadProc(__in  LPVOID lpParameter); 
const int THREAD_NUM = 10;CRITICAL_SECTION g_ThreadCode;int main() 
{HANDLE  handle[THREAD_NUM];    g_nNum = 0; int var = 0; InitializeCriticalSection(&g_ThreadCode); while ( var< THREAD_NUM) { handle[ var++] = CreateThread(NULL, 0, ThreadProc, NULL, 0, NULL); } WaitForMultipleObjects(THREAD_NUM, handle, TRUE, INFINITE); DeleteCriticalSection( &g_ThreadCode ); for( var=0; var<sizeof(handle); var++) { CloseHandle(handle[var]); }return 0; 
}DWORD WINAPI ThreadProc(__in  LPVOID lpParameter) 
{  EnterCriticalSection( &g_ThreadCode ); g_nNum++;  printf("当前计数为:%d\n",g_nNum); LeaveCriticalSection( &g_ThreadCode ); return 0; 
}

image

转载于:https://www.cnblogs.com/staring-hxs/p/3664634.html

这篇关于window下线程同步之(Critical Sections(关键代码段、关键区域、临界区域)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Spring Boot3虚拟线程的使用步骤详解

《SpringBoot3虚拟线程的使用步骤详解》虚拟线程是Java19中引入的一个新特性,旨在通过简化线程管理来提升应用程序的并发性能,:本文主要介绍SpringBoot3虚拟线程的使用步骤,... 目录问题根源分析解决方案验证验证实验实验1:未启用keep-alive实验2:启用keep-alive扩展建

Python 中的异步与同步深度解析(实践记录)

《Python中的异步与同步深度解析(实践记录)》在Python编程世界里,异步和同步的概念是理解程序执行流程和性能优化的关键,这篇文章将带你深入了解它们的差异,以及阻塞和非阻塞的特性,同时通过实际... 目录python中的异步与同步:深度解析与实践异步与同步的定义异步同步阻塞与非阻塞的概念阻塞非阻塞同步

如何使用Python实现一个简单的window任务管理器

《如何使用Python实现一个简单的window任务管理器》这篇文章主要为大家详细介绍了如何使用Python实现一个简单的window任务管理器,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起... 任务管理器效果图完整代码import tkinter as tkfrom tkinter i

Java终止正在运行的线程的三种方法

《Java终止正在运行的线程的三种方法》停止一个线程意味着在任务处理完任务之前停掉正在做的操作,也就是放弃当前的操作,停止一个线程可以用Thread.stop()方法,但最好不要用它,本文给大家介绍了... 目录前言1. 停止不了的线程2. 判断线程是否停止状态3. 能停止的线程–异常法4. 在沉睡中停止5

Java捕获ThreadPoolExecutor内部线程异常的四种方法

《Java捕获ThreadPoolExecutor内部线程异常的四种方法》这篇文章主要为大家详细介绍了Java捕获ThreadPoolExecutor内部线程异常的四种方法,文中的示例代码讲解详细,感... 目录方案 1方案 2方案 3方案 4结论方案 1使用 execute + try-catch 记录

Linux搭建Mysql主从同步的教程

《Linux搭建Mysql主从同步的教程》:本文主要介绍Linux搭建Mysql主从同步的教程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux搭建mysql主从同步1.启动mysql服务2.修改Mysql主库配置文件/etc/my.cnf3.重启主库my

Spring Boot 中正确地在异步线程中使用 HttpServletRequest的方法

《SpringBoot中正确地在异步线程中使用HttpServletRequest的方法》文章讨论了在SpringBoot中如何在异步线程中正确使用HttpServletRequest的问题,... 目录前言一、问题的来源:为什么异步线程中无法访问 HttpServletRequest?1. 请求上下文与线

在 Spring Boot 中使用异步线程时的 HttpServletRequest 复用问题记录

《在SpringBoot中使用异步线程时的HttpServletRequest复用问题记录》文章讨论了在SpringBoot中使用异步线程时,由于HttpServletRequest复用导致... 目录一、问题描述:异步线程操作导致请求复用时 Cookie 解析失败1. 场景背景2. 问题根源二、问题详细分

Java中将异步调用转为同步的五种实现方法

《Java中将异步调用转为同步的五种实现方法》本文介绍了将异步调用转为同步阻塞模式的五种方法:wait/notify、ReentrantLock+Condition、Future、CountDownL... 目录异步与同步的核心区别方法一:使用wait/notify + synchronized代码示例关键

Java多线程父线程向子线程传值问题及解决

《Java多线程父线程向子线程传值问题及解决》文章总结了5种解决父子之间数据传递困扰的解决方案,包括ThreadLocal+TaskDecorator、UserUtils、CustomTaskDeco... 目录1 背景2 ThreadLocal+TaskDecorator3 RequestContextH