为什么要对多线程进行加锁操作呢

一.如果不加锁,会怎么样?

可能会发生数据竞争,造成数据错乱.

例子:

本来想要的结果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;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

CodingLife99

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值