七、C++锁的使用

  1. 互斥锁
  2. 条件变量
  3. 原子操作

一、互斥锁

1.互斥锁(std::mutex)

使用std::lock_guard、std::unique_lock方便对线程进行加锁解锁,对std::mutex进行维护使用,不用再手动进行上锁加锁。

2.std::lock_guard

C++11提出,维护mutex的模板,自动管理互斥锁,构造函数内对锁进行加锁,析构时进行解锁,避免出现死锁、资源泄漏的情况。

3.std::unique_lock

在std::lock_guard基础上,可以手动对锁进行操作,std::unique_lock比std::lock_guard灵活,但是效率低,占用内存多点

#include "QT_mutex.h"
#include <QDebug>
#include <thread>
#include <string>
#include <iostream>
QT_mutex::QT_mutex(QWidget *parent)
    : QMainWindow(parent)
{
    ui.setupUi(this);
	std::thread thread1[3];
	for (int i = 0; i < 3; i++)
	{
		thread1[i] = std::thread(&QT_mutex::func1,this);
	}
	

	std::thread thread2[3];
	for (int i = 0; i < 3; i++)
	{
		thread2[i] = std::thread(&QT_mutex::func2, this);
	}
	
	
	std::thread thread3[3];
	for (int i = 0; i < 3; i++)
	{
		thread3[i] = std::thread(&QT_mutex::func3, this);
	}
	for (std::thread& th : thread3)
	{
		th.join();
	}
	for (std::thread& th : thread1)
	{
		th.join();
	}
	for (std::thread& th : thread2)
	{
		th.join();
	}
	qDebug() << "10 threads end";
}

void QT_mutex::func1()
{
	for (int i = 0; i < 100; ++i)
	{
		mutex1.lock();
		++j;
		std::thread::id id = std::this_thread::get_id();
		std::cout <<"func1" << " " <<id << " " << j << endl;
		mutex1.unlock();
	}
}

void QT_mutex::func2()
{
	for (int i = 0; i < 100; ++i)
	{
		std::lock_guard<std::mutex> lock(mutex2);
		++j2;
		std::thread::id id = std::this_thread::get_id();
		std::cout << "func2" << " " << id << " " << j2 << endl;
	}
}

void QT_mutex::func3()
{
	for (int i = 0; i < 100; ++i)
	{
		std::unique_lock<std::mutex> lock(mutex3);
		/*
		lock.unlock();//解锁
		//
		lock.lock();//加锁
		//
		++j3;
		if (lock.try_lock())//尝试加锁
		{
			//加锁成功
		}*/
		std::thread::id id = std::this_thread::get_id();
		std::cout << "func3" <<" "<< id << " " << j3 << endl;
	}
}

QT_mutex::~QT_mutex()
{}
#pragma once

#include <QtWidgets/QMainWindow>
#include "ui_QT_mutex.h"
#include <mutex>
using namespace std;

class QT_mutex : public QMainWindow
{
    Q_OBJECT

public:
    QT_mutex(QWidget *parent = nullptr);
	void func1();
	void func2();
	void func3();

    ~QT_mutex();

private:
    Ui::QT_mutexClass ui;
	std::mutex mutex1, mutex2, mutex3;
	int j = 0,j2 = 0,j3 = 0;

};

二.条件变量

condition_variable

1.condition_variable.wait(std::unique_lock<std::mutex>& lock)

当前线程的执行会被阻塞,直到收到 notify 为止

template<class Predicate>

condition_variable.wait(std::unique_lock<std::mutex>& lock,Predicate pred)

pred=false时导致当前线程阻塞并循环直到未true,pred=true时唤醒

2.condition_variable.notify_all()通知所有

condition_variable.notify_one()通知一个

int main()
{
    std::thread thread1[3];
	for (int i = 0; i < 3; i++)
	{
		thread1[i] = std::thread(&QT_mutex::threadfunc,this, i);

	}
	std::this_thread::sleep_for(std::chrono::seconds(2));
	// 修改共享状态,通知等待的线程
	//std::lock_guard<std::mutex> lock(mutex4);
	THREAD_FLAG = true;
	std::cout << "Notifying all threads...\n";
	cv.notify_all();  // 通知所有等待的线程
	for (int i = 0; i < 3; i++)
	{
		thread1[i].join();

	}
}

void QT_mutex::threadfunc(int i)
{
	std::unique_lock<std::mutex> lock(mutex4);
	if (!THREAD_FLAG)
	{
		cout <<i<< " wait cv" << endl;
		cv.wait(lock);
	}
	cout<<i << " threadfunc" << endl;
}

三.原子操作(atomic)

std::atomic<int> m_count(0);
void main()
{
    std::thread t1(&QT_mutex::threadatomic,this);
	std::thread t2(&QT_mutex::threadatomic,this);
	t1.join();
	t2.join();
	std::cout << "Counter value: " << m_count << std::endl;//20000
}

void QT_mutex::threadatomic()
{
	for (int i = 0; i < 10000; ++i) {
		m_count++; // 原子递增操作
	}
}

  • 15
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值