C++多线程4-unique_lock详解

unique_lock和lock_guard都是可以自动解锁的类,但是lock_guard更加高效体现在永远在析构函数中解锁,而unique_lock更加灵活,但执行效率会比lock_guard低一些,它们都是有不同的应用场景。
unique_lock是通用互斥包装器,允许延迟锁定、尝试锁定、递归锁定、所有权转移和条件变量一同使用;它可以移动mutex的所有权,但不可复制;
1.unique_lock的行为,通过第二个参数控制:
a)std::adopt_lock
在unique_lock创建对象,有adopt_lock参数时,对象不能执行lock成员函数,因为它的使用场景是mutex已经加了锁。

//测试使用adopt_lock参数创建的对象调用lock函数
#include 
#include 
#include 

using namespace std;

int main()
{
	mutex mutex_m;
	unique_lock<mutex> lock(mutex_m, std::adopt_lock);
	lock.lock();
	return 0;
}
//打印结果
terminate called after throwing an instance of 'std::system_error'
  what():  Resource deadlock avoided
Aborted (core dumped)

b)std::try_to_lock
在unique_lock创建对象,有try_to_lock参数时,对象会尝试去加锁,如果不成功,则会返回false,而不用继续等待。

#include 
#include 
#include 

using namespace std;

class Test
{
public:
	Test():age_m(0)
	{
		cout << "Test构造函数" << endl;
	}

	~Test()
	{
		cout << "Test析构函数" << endl;
	}

	void set(const int age)
	{
		std::lock_guard<std::mutex> lockgd(mutex_m);
		std::this_thread::sleep_for(std::chrono::seconds(2));
		cout << "set ,thread id:" << std::this_thread::get_id()<< endl;
		age_m = age;
	}

	void get()
	{
		std::unique_lock<std::mutex> unilock(mutex_m, std::try_to_lock);
		if (unilock.owns_lock())
		{
			cout << "get lock, thread id:" << std::this_thread::get_id()<< endl;
			cout << "age:" << age_m << endl;
		}
		else
		{
			cout << "Cant get lock, thread id:" << std::this_thread::get_id()<< endl;
		}
	}

private:
	int age_m;
	static std::mutex mutex_m;
};

std::mutex Test::mutex_m;

int main()
{
	Test test;
	std::thread th[20];
//	std::thread first(&Test::set, &test, 20);
//	first.join();

	for (int i=0; i<20; i++)
	{
		if (i%20 == 0)
			th[i] = std::thread(&Test::set, &test, i);
		else
			th[i] = std::thread(&Test::get, &test);
	}

	for (int i=0; i<20; i++)
	{
		if (th[i].joinable())
		{
			th[i].join();
		}
	}
	return 0;
}
//打印结果
root@epc:/home/share/eclipse_workspace/exercise/src#  ./exercise
Test构造函数
get lock, thread id:139643157436160
age:0
Cant get lock, thread id:139643149043456
Cant get lock, thread id:139643140650752
Cant get lock, thread id:139643165828864
get lock, thread id:139643132258048
age:0
Cant get lock, thread id:139643123865344
Cant get lock, thread id:139643115472640
get lock, thread id:139643174221568
age:0
Cant get lock, thread id:139643107079936
get lock, thread id:139643098687232
age:0
get lock, thread id:139643090294528
age:0
get lock, thread id:139643081901824
age:0
Cant get lock, thread id:139643182614272
get lock, thread id:139643073509120
age:0
get lock, thread id:139643065116416
age:0
get lock, thread id:139643056723712
age:0
get lock, thread id:139643048331008
age:0
get lock, thread id:139643191006976
age:0
get lock, thread id:139643199399680
age:0
set ,thread id:139643207792384
Test析构函数

c)std::defer_lock
在unique_lock创建对象,有defer_lock参数时,需要调用成员函数lock进行加锁;

class Test
{
public:
	Test():age_m(0)
	{
		cout << "Test构造函数" << endl;
	}

	~Test()
	{
		cout << "Test析构函数" << endl;
	}

	void set(const int age)
	{
		std::lock_guard<std::mutex> lockgd(mutex_m);
//		std::this_thread::sleep_for(std::chrono::seconds(1));
		cout << "set age= " << age << " ,thread id:" << std::this_thread::get_id()<< endl;
		age_m = age;
	}

	void get()
	{
		std::unique_lock<std::mutex> unilock(mutex_m, std::defer_lock);
		if (unilock.try_lock())
		{
			cout << "get age=" << age_m <<  ", get lock, thread id:" << std::this_thread::get_id()<< endl;
		}
		else
		{
			cout << "Cant get lock, thread id:" << std::this_thread::get_id()<< endl;
		}
	}

private:
	int age_m;
	static std::mutex mutex_m;
};

std::mutex Test::mutex_m;

2.unique_lock成员函数:
a)lock
在unique_lock创建对象,有defer_lock参数时,需要调用成员函数lock进行加锁;
b)unlock
可以利用unlock成员函数来控制加锁代码的粒度;
c)try_lock
在unique_lock创建对象,有defer_lock参数时,可以利用成员函数try_lock尝试加锁,成功则返回true,不成功则返回false;
d)release
执行完release后,将mutex的所有权释放,并返回mutex的对象指针;
举例:

std::mutex* mutex_p = unilock.release();
if (mutex_p != nullptr) mutex_p->unlock();

3.unique的所有权传递
unique_lock不支持拷贝和赋值,
举例:

std::unique_lock<std::mutex> templock(mutex_m, std::defer_lock);
std::unique_lock<std::mutex> unilock(move(templock));

你可能感兴趣的:(C++编程)