互斥锁、读写锁和自旋锁的使用场景
时间: 2024-06-06 12:05:46 浏览: 26
互斥锁、读写锁和自旋锁都是用来保护共享资源的并发访问的。它们的使用场景如下:
1. 互斥锁:适用于保护临界区,即一次只能有一个线程访问的共享资源。当一个线程获得互斥锁时,其他线程会被阻塞,直到该线程释放锁。互斥锁适用于读写操作不频繁、临界区较小的情况。
2. 读写锁:适用于读操作比写操作频繁的场景。读写锁允许多个线程同时读取共享资源,但在写操作时必须独占锁。当有写操作时,所有读操作都会被阻塞,直到写操作完成。读写锁适用于读操作占比较大的场景,可以提高并发性能。
3. 自旋锁:适用于保护临界区,且临界区很小的场景。自旋锁不会使线程阻塞,而是在获取锁失败时,不停地尝试获取锁,直到获取成功。自旋锁适用于临界区很小,线程获取锁的时间很短的场景,可以避免线程上下文切换的开销。
相关问题
请给出互斥锁、读写锁以及自旋锁和递归锁的用法实例
1. 互斥锁
互斥锁用于保护共享资源,确保同一时刻只有一个线程能够访问该资源。实例代码如下:
```C++
#include <mutex>
#include <thread>
#include <iostream>
std::mutex mtx;
void print_func(int num) {
mtx.lock();
std::cout << "Thread " << num << " is printing." << std::endl;
mtx.unlock();
}
int main() {
std::thread t1(print_func, 1);
std::thread t2(print_func, 2);
t1.join();
t2.join();
return 0;
}
```
2. 读写锁
读写锁用于在读多写少的场景中提高效率,允许多个线程同时读取共享资源,但只允许一个线程写入共享资源。实例代码如下:
```C++
#include <iostream>
#include <thread>
#include <shared_mutex>
std::shared_mutex rw_mutex;
int shared_data = 0;
void read_func(int num) {
rw_mutex.lock_shared();
std::cout << "Thread " << num << " read shared_data: " << shared_data << std::endl;
rw_mutex.unlock_shared();
}
void write_func(int num) {
rw_mutex.lock();
shared_data = num;
std::cout << "Thread " << num << " wrote shared_data: " << shared_data << std::endl;
rw_mutex.unlock();
}
int main() {
std::thread t1(write_func, 1);
std::thread t2(read_func, 2);
std::thread t3(read_func, 3);
t1.join();
t2.join();
t3.join();
return 0;
}
```
3. 自旋锁
自旋锁适用于短时间内需要等待的场景,可以减少线程切换带来的开销。实例代码如下:
```C++
#include <iostream>
#include <thread>
#include <atomic>
std::atomic_flag spin_lock = ATOMIC_FLAG_INIT;
int shared_data = 0;
void add_func(int num) {
while (spin_lock.test_and_set(std::memory_order_acquire));
shared_data += num;
spin_lock.clear(std::memory_order_release);
}
int main() {
std::thread t1(add_func, 1);
std::thread t2(add_func, 2);
t1.join();
t2.join();
std::cout << "shared_data: " << shared_data << std::endl;
return 0;
}
```
4. 递归锁
递归锁允许同一线程多次获取锁,避免死锁。实例代码如下:
```C++
#include <iostream>
#include <thread>
#include <mutex>
std::recursive_mutex mtx;
int shared_data = 0;
void add_func(int num) {
mtx.lock();
shared_data += num;
if (num > 1) {
add_func(num - 1);
}
mtx.unlock();
}
int main() {
std::thread t1(add_func, 3);
t1.join();
std::cout << "shared_data: " << shared_data << std::endl;
return 0;
}
```
详细介绍多线程加锁有三种方式:互斥锁、读写锁和自旋锁。
好的,我可以回答这个问题。多线程加锁通常有三种方式:互斥锁、读写锁和自旋锁。互斥锁是一种最常用的锁,它可以确保同一时间只有一个线程可以访问共享资源。读写锁则可以允许多个线程同时读取共享资源,但是只允许一个线程进行写操作。自旋锁则是一种特殊的锁,它会在线程需要访问共享资源时进行忙等待,直到锁被释放为止。这三种锁的应用场景不同,需要根据实际情况选择合适的锁来保证程序的正确性和效率。
阅读全文