标签:加锁 int include Num 线程 ThreadNum 操作 多线程
一.如果不加锁,会怎么样?
可能会发生数据竞争,造成数据错乱.
例子:
本来想要的结果n=0,但是执行发现n的值不为0,而且有多种取值.究其原因,是因为多个线程之间会发生数据竞争,导致CPU线程调度时出现问题,不能够保证线程内执行代码的原子操作.我发现string str = "hello";这一句是必要的,不然它就不会出现n不为0的现象.(有待进一步研究)
注意,如果要所有子线程执行完毕后,再执行主线程.要有WaitForMultipleObjects操作.
#include "windows.h" #include <iostream>using namespace std;const int ThreadNum = 50; int g_Num = 0; DWORD WINAPI ThreadFun(void * param) { for (int i = 0; i < 100; i++) { g_Num = g_Num+1; string str = "hello"; g_Num = g_Num-1; } return 0; } int main() { DWORD lpThread[ThreadNum]; HANDLE h[ThreadNum]; for (int i = 0; i <ThreadNum; i++) { h[i] = CreateThread(NULL, 0, ThreadFun, NULL, 0, &lpThread[i]); } WaitForMultipleObjects(ThreadNum, h, TRUE, INFINITE); //等待所有的子线程执行完毕. int n = g_Num; cout << "n:" << n << endl; return 0; }
二.加锁后的情形
WaitForMultipleObjects是必须的,一开始忘记了使用这个,导致在EnterCriticalSection处报错,猜测是主线程没有等子线程执行完毕就结束了程序.为了验证猜想,将WaitForMultipleObjects注释掉,
在子函数中打印,发现确实是这个原因.
加锁后每次n打印出来的都是0了.
#include "windows.h" #include <iostream> using namespace std; const int ThreadNum = 50; CRITICAL_SECTION g_Cs; int g_Num = 0; DWORD WINAPI ThreadFun(void * param) { for (int i = 0; i < 100; i++) { EnterCriticalSection(&g_Cs); g_Num = g_Num+1; string str = "hello"; g_Num = g_Num-1; LeaveCriticalSection(&g_Cs); } return 0; } int main() { InitializeCriticalSection(&g_Cs); DWORD lpThread[ThreadNum]; HANDLE h[ThreadNum]; for (int i = 0; i <ThreadNum; i++) { h[i] = CreateThread(NULL, 0, ThreadFun, NULL, 0, &lpThread[i]); } WaitForMultipleObjects(ThreadNum, h, TRUE, INFINITE); //等待所有的子线程执行完毕. int n = g_Num; cout << "n:" << n << endl; DeleteCriticalSection(&g_Cs); return 0; }
标签:加锁,int,include,Num,线程,ThreadNum,操作,多线程
来源: https://www.cnblogs.com/Stephen-Qin/p/12702487.html
本站声明:
1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。