# 线程的同步和互斥
在多线程编程中,共享资源的使用是提高程序运行效率的关键,但同时也带来了一系列挑战。由于线程间缺乏隔离机制,共享同一内存地址的操作可能导致所谓的 “竞争条件”,这是指多个线程同时访问并试图修改同一资源时可能发生的问题。这种情况会导致程序的执行结果与预期出现显著偏差。
例如,在以下代码示例中,两个线程被设计为对一个共享变量进行递增操作,每个线程执行一百万次。然而,实际执行结果往往与预期的两百万次递增不符。
#include <pthread.h> | |
#define TIMES 1000000 | |
int global = 0; | |
void *increment(void *arg){ | |
printf("Child thread is running.\n"); | |
for(int i = 0; i < TIMES; i++){ | |
global++; | |
} | |
printf("Child thread has finished.\n"); | |
return NULL; | |
} | |
int main(int argc, char *argv[]){ | |
pthread_t tid; | |
pthread_create(&tid, NULL, increment, NULL); | |
printf("Main thread is running.\n"); | |
for(int i = 0; i < TIMES; i++){ | |
global++; | |
} | |
printf("Main thread has finished.\n"); | |
pthread_join(tid, NULL); | |
printf("Final result: global = %d\n", global); | |
return 0; | |
} |
结果的偏差源于线程间对共享变量的非原子性访问。例如,当主线程读取共享变量的值时,它可能会将该值从内存加载到寄存器,增加 1 后再尝试写回内存。然而,在写回过程中,如果子线程也读取了该变量的值(此时尚未更新),并执行了相同的增加操作,最终的结果就是两个线程的操作相互覆盖,导致共享变量的值只增加了 1,而非预期的两百万。
为了确保程序的正确性,即使在高效率的并发环境下,也需要使用线程库提供的同步机制,如锁,来正确地管理对共享资源的访问。通过使用锁,可以保证在任何时刻只有一个线程能够访问共享资源,从而避免了竞争条件的发生,确保了程序结果的准确性和可靠性。
# 互斥锁
在多线程编程实践中,互斥锁(MUTual EXclusion,简称 mutex)是一种基础且广泛使用的同步机制。它是一种用于管理共享资源访问的机制,通过一个全局的布尔标志位来实现。线程能够对这一标志位进行原子性的加锁和解锁操作。当一个线程成功加锁时,任何其他线程的加锁尝试都将导致它们进入阻塞状态,直到锁被当前持有者解锁。解锁后,其他线程便有机会获得这把锁,并从等待状态中恢复执行。这种机制确保了在任何给定时间点,锁不会被两个线程同时持有,从而避免了资源的并发访问冲突。
# 锁的基本情况
# 互斥锁(Mutex)
互斥锁是一种关键的同步工具,用于在多线程环境中协调对共享资源的访问。它的核心作用是保证在任何给定时刻,只有一个线程能够执行对共享资源的操作,从而避免了并发访问带来的问题。
# 锁的状态
未锁定:在这种状态下,没有线程持有锁,任何线程都可以尝试获取锁。
锁定:当一个线程成功获取锁后,其他线程的获取尝试将导致它们进入等待状态,直到锁被释放。
# 锁的行为
加锁:这是一个原子操作,检查锁的状态。如果锁未被锁定,线程将获取锁;如果锁已被锁定,线程将被阻塞。
解锁:这个操作将锁设置为未锁定状态,允许其他等待的线程尝试获取锁。
# 锁的要求
- 确保在任何时刻只有一个线程能够获取锁。
- 遵循 “谁加锁,谁解锁” 的原则,以避免潜在的错误和代码的不可读性。
# 临界区
临界区是指在加锁和解锁之间的代码段,这段代码可以安全地访问共享资源。由于互斥锁的互斥性,可以确保在临界区内只有一个线程能够执行。
# 饥饿
饥饿是指线程由于无法获得所需的锁而长时间处于等待状态。这可能是由于锁的竞争,或者临界区过大导致的。饥饿可能会影响程序的性能和稳定性。
# 死锁
死锁是一种严重的同步问题,发生在线程由于不当的资源竞争而永久阻塞。在使用互斥锁时,必须小心避免死锁,特别是当需要获取多个锁时,应保持一致的加锁顺序。
# 常见的死锁情况
- 循环等待:两个或多个线程分别持有对方需要的锁,导致无法进一步执行。
- 锁的非释放:持有锁的线程在未释放锁的情况下终止,导致其他线程无法获取该锁。
- 重复加锁:一个线程在持有锁的情况下,尝试再次获取同一把锁,这可能导致死锁。
# 锁的基本使用
# 定义锁
首先,需要定义一个互斥锁(mutex),通常使用 pthread_mutex_t
类型。例如:
pthread_mutex_t mLock; |
# 初始化锁
锁的初始化是使用前的必要步骤,可以通过以下两种方式之一进行:
-
使用
pthread_mutex_init
函数: 调用此函数可以初始化锁,并可选地指定锁的属性。pthread_mutex_init
是 POSIX 线程库中用于初始化互斥锁(mutex)的函数。互斥锁是一种同步机制,用于保护共享资源不被多个线程同时访问,从而避免竞态条件和数据不一致的问题。函数原型如下:
#include <pthread.h>
int pthread_mutex_init(pthread_mutex_t *mutex, const pthread_mutexattr_t *attr);
参数说明:
- pthread_mutex_t *mutex: 这是一个指向
pthread_mutex_t
类型的指针,用于存储互斥锁的标识符。成功初始化后,这个标识符将被用来在后续操作中引用这个互斥锁。 - const pthread_mutexattr_t *attr: 这是一个指向
pthread_mutexattr_t
类型的指针,它指定了互斥锁的属性。如果这个参数是NULL
,互斥锁将被初始化为默认属性。
返回值:
- 如果
pthread_mutex_init
调用成功,返回 0。 - 如果调用失败,返回一个错误码。例如,如果内存分配失败,将返回
ENOMEM
。
使用场景:
- 当需要在多线程程序中保护共享数据或资源时。
- 当需要确保某个操作或代码段在同一时间只被一个线程执行时。
工作机制:
- 通过调用
pthread_mutex_init
,操作系统会分配必要的资源来创建一个互斥锁,并根据提供的属性(如果有)初始化它。 - 初始化后,互斥锁处于未锁定状态。
- pthread_mutex_t *mutex: 这是一个指向
-
使用宏
PTHREAD_MUTEX_INITIALIZER
: 这是一种简便的初始化方式,适用于静态初始化。pthread_mutex_t mLock = PTHREAD_MUTEX_INITIALIZER;
# 加锁
在进入临界区之前,使用 pthread_mutex_lock
函数来获取锁。如果锁已经被其他线程持有,当前线程将被阻塞,直到锁被释放。
pthread_mutex_lock
是 POSIX 线程库中用于请求互斥锁的函数。当一个线程需要访问共享资源时,它会首先尝试锁定互斥锁,以确保在访问期间资源不会被其他线程修改。
函数原型如下:
#include <pthread.h> | |
int pthread_mutex_lock(pthread_mutex_t *mutex); |
参数说明:
- pthread_mutex_t *mutex: 这是一个指向
pthread_mutex_t
类型的指针,它标识了要锁定的互斥锁。这个互斥锁必须已经通过pthread_mutex_init
函数初始化。
返回值:
- 如果
pthread_mutex_lock
调用成功,返回 0。 - 如果调用失败,返回一个错误码。例如:
EBUSY
: 互斥锁已经被其他线程锁定。EDEADLK
: 死锁被避免,因为线程试图锁定一个它已经持有的互斥锁(这取决于互斥锁的属性)。
使用场景:
- 当需要保护共享资源,确保同一时间只有一个线程可以访问这些资源。
- 在进入临界区(critical section)之前,线程需要调用
pthread_mutex_lock
来锁定互斥锁。
工作机制:
- 当一个线程调用
pthread_mutex_lock
时,如果互斥锁未被其他线程锁定,该函数会立即锁定互斥锁并返回。 - 如果互斥锁已经被其他线程锁定,调用
pthread_mutex_lock
的线程将阻塞,直到互斥锁被释放。
# 解锁
在临界区的代码执行完毕后,使用 pthread_mutex_unlock
函数释放锁,允许其他线程进入临界区。
pthread_mutex_unlock
是 POSIX 线程库中用于释放互斥锁的函数。当一个线程完成了对共享资源的访问后,它会调用这个函数来解锁互斥锁,允许其他正在等待这个锁的线程访问资源。
函数原型如下:
#include <pthread.h> | |
int pthread_mutex_unlock(pthread_mutex_t *mutex); |
参数说明:pthread_mutex_t *mutex: 这是一个指向 pthread_mutex_t
类型的指针,它标识了要释放的互斥锁。这个互斥锁必须已经通过 pthread_mutex_init
函数初始化,并且当前线程必须已经拥有这个锁。
返回值:
- 如果
pthread_mutex_unlock
调用成功,返回 0。 - 如果调用失败,返回一个错误码。例如:
EINVAL
: 传入的互斥锁参数无效或未初始化。EPERM
: 调用线程不拥有互斥锁的所有权,或者互斥锁被设为了错误检测状态。
使用场景:
- 当线程完成了对共享资源的访问,需要释放互斥锁以允许其他线程访问资源。
- 在临界区代码执行完毕后,线程需要调用
pthread_mutex_unlock
来释放互斥锁。
工作机制:
- 当一个线程调用
pthread_mutex_unlock
时,它将减少互斥锁的锁定计数。如果锁定计数达到零,互斥锁将变为未锁定状态,其他等待这个锁的线程将有机会获取它。 - 如果互斥锁的属性设置为错误检测,并且当前线程不拥有互斥锁,调用
pthread_mutex_unlock
将失败并返回EPERM
错误。
# 销毁锁
在不再需要锁时,使用 pthread_mutex_destroy
函数销毁它,以释放系统资源。
pthread_mutex_destroy
是 POSIX 线程库中用于销毁互斥锁的函数。互斥锁(mutex)是用来协调对共享资源访问的同步机制,确保在同一时刻只有一个线程可以访问特定的资源。当互斥锁不再需要时,比如在程序结束或某个模块卸载时,应该使用 pthread_mutex_destroy
来释放与互斥锁关联的资源。
函数原型如下:
#include <pthread.h> | |
int pthread_mutex_destroy(pthread_mutex_t *mutex); |
参数说明:
- pthread_mutex_t *mutex: 这是一个指向
pthread_mutex_t
类型的指针,它标识了要销毁的互斥锁。这个互斥锁必须已经通过pthread_mutex_init
函数初始化。
返回值:
- 如果
pthread_mutex_destroy
调用成功,返回 0。 - 如果调用失败,返回一个错误码。例如:
EBUSY
: 互斥锁仍然被锁定,或者有其他线程正在等待这个锁。EINVAL
: 传入的互斥锁参数无效或未初始化。
使用场景:
- 当互斥锁不再被使用,需要释放相关资源时。
- 在程序模块卸载或线程结束前,确保所有互斥锁都被销毁。
工作机制:
- 当调用
pthread_mutex_destroy
时,操作系统会检查互斥锁是否处于未锁定状态。如果互斥锁被锁定或有线程正在等待它,销毁操作将失败。 - 成功销毁互斥锁后,与该互斥锁关联的所有资源将被释放。
# 使用案例
案例一:基本用法
#include <pthread.h> | |
#define TIMES 1000000 | |
int global = 0; | |
// 子线程函数 | |
void *increment(void *arg){ | |
pthread_mutex_t *pmLock = (pthread_mutex_t *)arg; // 从主线程接收锁 | |
printf("Child thread is running.\n"); | |
for(int i = 0; i < TIMES; i++){ | |
pthread_mutex_lock(pmLock); // 加锁 | |
global++; | |
pthread_mutex_unlock(pmLock); // 解锁 | |
} | |
printf("Child thread has finished.\n"); | |
return NULL; | |
} | |
int main(int argc, char *argv[]){ | |
pthread_mutex_t mLock; // 定义互斥锁 | |
pthread_mutex_init(&mLock, NULL); // 初始化互斥锁 | |
pthread_t pid; | |
pthread_create(&pid, NULL, increment, &mLock); // 将互斥锁传递给子线程 | |
printf("Main thread is running.\n"); | |
for(int i = 0; i < TIMES; i++){ | |
pthread_mutex_lock(&mLock); // 加锁 | |
global++; | |
pthread_mutex_unlock(&mLock); // 解锁 | |
} | |
printf("Main thread has finished.\n"); | |
pthread_join(pid, NULL); // 等待子线程结束 | |
printf("All operations completed, global = %d\n", global); | |
pthread_mutex_destroy(&mLock); // 销毁互斥锁 | |
return 0; | |
} |
- 定义和初始化互斥锁:在
main
函数中,首先定义了一个pthread_mutex_t
类型的互斥锁,并使用pthread_mutex_init
函数对其进行初始化。 - 创建子线程:使用
pthread_create
函数创建了一个子线程,并将互斥锁的地址作为参数传递给子线程的函数increment
。 - 加锁和解锁:在
increment
函数和main
函数中,每次对共享变量global
进行操作之前,都会先通过pthread_mutex_lock
函数加锁,操作完成后通过pthread_mutex_unlock
函数解锁。这样可以确保在对共享资源进行修改时,只有一个线程能够执行该操作,从而避免了竞争条件。 - 等待子线程结束:使用
pthread_join
函数等待子线程结束,确保在主线程继续执行之前子线程已经完成所有操作。 - 销毁互斥锁:在程序结束前,使用
pthread_mutex_destroy
函数销毁互斥锁,释放系统资源。
示例二:使用 2 个线程对同一个全局变量各加 2000 万
#include <pthread.h> | |
#include <stdio.h> | |
#include <sys/time.h> | |
#define TIMES 10000000 | |
// 定义共享数据结构 | |
typedef struct { | |
int sum; | |
pthread_mutex_t mLock; | |
} share_value_t; | |
// 子线程函数 | |
void *increment(void *arg) { | |
share_value_t *pshareValue = (share_value_t *)arg; | |
printf("Child thread is running.\n"); | |
for (int i = 0; i < TIMES; i++) { | |
pthread_mutex_lock(&pshareValue->mLock); | |
pshareValue->sum++; | |
pthread_mutex_unlock(&pshareValue->mLock); | |
} | |
printf("Child thread has finished.\n"); | |
return NULL; | |
} | |
int main(int argc, char *argv[]) { | |
struct timeval beginTime, endTime; | |
gettimeofday(&beginTime, NULL); // 获取开始时间 | |
share_value_t shareValue; | |
shareValue.sum = 0; | |
pthread_mutex_init(&shareValue.mLock, NULL); // 初始化互斥锁 | |
pthread_t pid; | |
pthread_create(&pid, NULL, increment, &shareValue); // 创建子线程 | |
printf("Main thread is running.\n"); | |
for (int i = 0; i < TIMES; i++) { | |
pthread_mutex_lock(&shareValue.mLock); | |
shareValue.sum++; | |
pthread_mutex_unlock(&shareValue.mLock); | |
} | |
printf("Main thread has finished.\n"); | |
pthread_join(pid, NULL); // 等待子线程结束 | |
printf("All operations completed, sum = %d\n", shareValue.sum); | |
pthread_mutex_destroy(&shareValue.mLock); // 销毁互斥锁 | |
gettimeofday(&endTime, NULL); // 获取结束时间 | |
// 计算并打印执行时间 | |
long duration = (endTime.tv_sec - beginTime.tv_sec) * 1000000 + | |
(endTime.tv_usec - beginTime.tv_usec); | |
printf("Execution time: %ld us\n", duration); | |
return 0; | |
} |
# pthread_mutex_trylock
锁定互斥锁
pthread_mutex_trylock
是 POSIX 线程库中提供的一个非阻塞函数,用于尝试锁定一个互斥锁(mutex)。与 pthread_mutex_lock
不同, pthread_mutex_trylock
函数不会使调用线程挂起等待互斥锁,如果互斥锁已经被其他线程锁定,它会立即返回一个错误码。
函数原型如下:
#include <pthread.h> | |
int pthread_mutex_trylock(pthread_mutex_t *mutex); |
参数说明:pthread_mutex_t *mutex: 这是一个指向 pthread_mutex_t
类型的指针,它标识了要尝试锁定的互斥锁。这个互斥锁必须已经通过 pthread_mutex_init
函数初始化。
返回值:
- 如果互斥锁被成功锁定,返回 0。
- 如果互斥锁已经被其他线程锁定,返回
EBUSY
错误码。 - 如果调用失败,返回其他错误码。例如:
EINVAL
: 传入的互斥锁参数无效或未初始化。
使用场景:
- 当线程需要尝试访问共享资源,但不想在互斥锁已被锁定时等待。
- 在需要快速响应锁状态变化的场景中,
pthread_mutex_trylock
提供了一种快速失败的机制。
工作机制:
- 当一个线程调用
pthread_mutex_trylock
时,如果互斥锁是未锁定状态,该函数会锁定互斥锁并返回 0,此时调用线程拥有互斥锁。 - 如果互斥锁已经被其他线程锁定,
pthread_mutex_trylock
会立即返回EBUSY
,而不会使调用线程挂起。
代码一:死锁示例
在第一个示例中,线程 A 和线程 B 以不同的顺序尝试获取两个锁,这可能导致死锁。如果线程 A 持有锁 1 并等待锁 2,而线程 B 持有锁 2 并等待锁 1,它们将永远等待对方释放锁,从而导致死锁。
#include <pthread.h> | |
#include <unistd.h> | |
#define TIMES 10000000 | |
typedef struct share_value{ | |
int sum; | |
pthread_mutex_t mLock1; | |
pthread_mutex_t mLock2; | |
} share_value_t; | |
void * func(void *arg){ | |
share_value_t *pshareValue = (share_value_t *)arg; | |
pthread_mutex_lock(&pshareValue->mLock2); | |
sleep(1); | |
pthread_mutex_lock(&pshareValue->mLock1); | |
printf("child thread runing \n"); | |
pthread_mutex_unlock(&pshareValue->mLock1); | |
pthread_mutex_unlock(&pshareValue->mLock2); | |
return NULL; | |
} | |
int main(int argc,char*argv[]) | |
{ | |
share_value_t shareValue; | |
shareValue.sum = 0; | |
pthread_mutex_init(&shareValue.mLock1, NULL); | |
pthread_mutex_init(&shareValue.mLock2, NULL); | |
pthread_t pid; | |
pthread_create(&pid,NULL,func, &shareValue); | |
pthread_mutex_lock(&shareValue.mLock1); | |
sleep(1); | |
pthread_mutex_lock(&shareValue.mLock2); | |
printf("main thread runing \n"); | |
pthread_mutex_unlock(&shareValue.mLock2); | |
pthread_mutex_unlock(&shareValue.mLock1); | |
pthread_join(pid, NULL); | |
pthread_mutex_destroy(&shareValue.mLock1); | |
pthread_mutex_destroy(&shareValue.mLock2); | |
return 0; | |
} |
代码二:使用 pthread_mutex_trylock
避免死锁
在第二个示例中,我们使用 pthread_mutex_trylock
函数来尝试获取锁,如果获取失败,线程将释放已持有的锁并重试。这种方法可以避免死锁,因为线程不会在持有一个锁的同时无限期地等待另一个锁。
#include <pthread.h> | |
#include <unistd.h> | |
#define TIMES 10000000 | |
typedef struct { | |
int sum; | |
pthread_mutex_t mLock1; | |
pthread_mutex_t mLock2; | |
} share_value_t; | |
void *func(void *arg){ | |
share_value_t *pshareValue = (share_value_t *)arg; | |
while(1){ | |
pthread_mutex_lock(&pshareValue->mLock2); | |
sleep(1); | |
if(pthread_mutex_trylock(&pshareValue->mLock1) == 0){ | |
printf("Child thread is running.\n"); | |
pthread_mutex_unlock(&pshareValue->mLock1); | |
pthread_mutex_unlock(&pshareValue->mLock2); | |
break; | |
} else { | |
pthread_mutex_unlock(&pshareValue->mLock2); | |
} | |
} | |
return NULL; | |
} | |
int main(int argc, char *argv[]){ | |
share_value_t shareValue; | |
shareValue.sum = 0; | |
pthread_mutex_init(&shareValue.mLock1, NULL); | |
pthread_mutex_init(&shareValue.mLock2, NULL); | |
pthread_t pid; | |
pthread_create(&pid, NULL, func, &shareValue); | |
pthread_mutex_lock(&shareValue.mLock1); | |
sleep(1); | |
pthread_mutex_lock(&shareValue.mLock2); | |
printf("Main thread is running.\n"); | |
pthread_mutex_unlock(&shareValue.mLock2); | |
pthread_mutex_unlock(&shareValue.mLock1); | |
pthread_join(pid, NULL); | |
pthread_mutex_destroy(&shareValue.mLock1); | |
pthread_mutex_destroy(&shareValue.mLock2); | |
return 0; | |
} |
# 自旋锁
自旋锁是一种特殊的锁机制,它与互斥锁( pthread_mutex_*
)不同,主要区别在于线程在尝试获取锁时的行为。互斥锁会导致不满足条件的线程进入睡眠状态,从而不会占用 CPU 资源。而自旋锁则会让线程在当前位置不断循环,检查锁是否已被释放,这会导致线程一直占用 CPU 资源,直到获取锁。
使用 pthread_mutex_trylock
结合 while
循环确实可以实现自旋锁的思想,但这种方式并不是真正的自旋锁。真正的自旋锁是由 pthread_spinlock_*
函数族提供的,它包括初始化、锁定、尝试锁定、解锁和销毁等操作。
#include <pthread.h> | |
//spin lock: spin-> 旋转 | |
pthread_spinlock_t mLock; // 锁类型 | |
int pthread_spin_init(pthread_spinlock_t *lock int pshared);// 初始化锁 | |
// PTHREAD_PROCESS_PRIVATE: 表示自旋锁仅用于同一进程的不同线程之间的同步。 | |
// PTHREAD_PROCESS_SHARED: 表示自旋锁可以用于不同进程之间的同步 (前提是这个锁变量位于某种共享内存区域中) | |
int pthread_spin_lock(pthread_spinlock_t *lock); // 自旋获取锁 | |
int pthread_spin_trylock(pthread_spinlock_t *lock); // 无法获得锁直接返回 | |
int pthread_spin_unlock(pthread_spinlock_t *lock); // 解锁 | |
int pthread_spin_destroy(pthread_spinlock_t *lock); // 销毁锁 |
- 定义自旋锁:使用
pthread_spinlock_t
类型定义自旋锁变量。 - 初始化自旋锁:使用
pthread_spin_init
函数初始化自旋锁,PTHREAD_PROCESS_PRIVATE
表示自旋锁仅用于同一进程的不同线程之间的同步。 - 自旋获取锁:使用
pthread_spin_lock
函数尝试获取自旋锁,如果无法立即获取,线程将在当前位置循环,直到能够获取锁。 - 尝试自旋获取锁:使用
pthread_spin_trylock
函数尝试获取自旋锁,如果无法获取,函数会立即返回,而不会进入自旋状态。 - 解锁:使用
pthread_spin_unlock
函数释放自旋锁,允许其他线程获取该锁。 - 销毁自旋锁:在不再需要自旋锁时,使用
pthread_spin_destroy
函数销毁它,释放系统资源。
示例:
#include <pthread.h> | |
#include <stdio.h> | |
#include <unistd.h> | |
pthread_spinlock_t mSpinLock; | |
void *func(void *arg){ | |
// 子线程睡眠较短时间,模拟工作 | |
sleep(1); | |
if (pthread_spin_trylock(&mSpinLock) == 0) { | |
printf("Child thread gets the lock.\n"); | |
pthread_spin_unlock(&mSpinLock); | |
} else { | |
printf("Child thread failed to get the lock.\n"); | |
} | |
return NULL; | |
} | |
int main(int argc, char *argv[]) { | |
// 初始化自旋锁 | |
if (pthread_spin_init(&mSpinLock, PTHREAD_PROCESS_PRIVATE) != 0) { | |
perror("Failed to initialize spinlock"); | |
return 1; | |
} | |
pthread_t pid; | |
if (pthread_create(&pid, NULL, func, NULL) != 0) { | |
perror("Failed to create thread"); | |
pthread_spin_destroy(&mSpinLock); | |
return 1; | |
} | |
// 主线程获取自旋锁 | |
if (pthread_spin_trylock(&mSpinLock) == 0) { | |
printf("Main thread gets the lock.\n"); | |
// 模拟长时间工作 | |
sleep(10); | |
pthread_spin_unlock(&mSpinLock); | |
printf("Main thread releases the lock.\n"); | |
} else { | |
printf("Main thread failed to get the lock.\n"); | |
} | |
// 等待子线程结束 | |
pthread_join(pid, NULL); | |
// 销毁自旋锁 | |
if (pthread_spin_destroy(&mSpinLock) != 0) { | |
perror("Failed to destroy spinlock"); | |
return 1; | |
} | |
return 0; | |
} |
# 读写锁
读锁特性:
- 读锁支持多个线程 并行访问 共享资源,只要没有线程请求写锁。
- 当一个或多个线程持有读锁时,其他线程可以继续获取读锁进行读取操作,但写锁请求将被 阻塞。
写锁特性:
- 写锁要求 独占访问 共享资源,即在任何时刻只能有一个线程持有写锁。
- 当线程请求写锁时,所有其他读锁和写锁的请求都将被 挂起,直到写锁被释放。
这种锁的设计允许在没有写操作时提高资源的并发访问能力,同时确保写操作的安全性。读写锁在数据库、文件系统和多线程应用程序中非常常见,特别是在读取操作远多于写入操作的场景中。
# 读写锁的使用示例
#include <pthread.h> | |
#include <stdio.h> | |
#include <unistd.h> | |
// 定义读写锁 | |
pthread_rwlock_t rLock = PTHREAD_RWLOCK_INITIALIZER; | |
// 子线程函数 | |
void *func(void *arg){ | |
// 获取读锁 | |
pthread_rwlock_rdlock(&rLock); | |
printf("Child thread %ld is reading, num = %d\n", pthread_self(), num); | |
sleep(1); // 模拟读取操作 | |
printf("Child thread %ld finished reading\n", pthread_self()); | |
// 释放读锁 | |
pthread_rwlock_unlock(&rLock); | |
return NULL; | |
} | |
int main(int argc, char *argv[]) { | |
int num = 10; // 共享资源 | |
pthread_t pid1, pid2, pid3; | |
// 创建三个线程 | |
if (pthread_create(&pid1, NULL, func, NULL) != 0) { | |
perror("Failed to create thread 1"); | |
return 1; | |
} | |
if (pthread_create(&pid2, NULL, func, NULL) != 0) { | |
perror("Failed to create thread 2"); | |
return 1; | |
} | |
if (pthread_create(&pid3, NULL, func, NULL) != 0) { | |
perror("Failed to create thread 3"); | |
return 1; | |
} | |
// 等待所有线程完成 | |
pthread_join(pid1, NULL); | |
pthread_join(pid2, NULL); | |
pthread_join(pid3, NULL); | |
// 销毁读写锁 | |
pthread_rwlock_destroy(&rLock); | |
return 0; | |
} |
解释:
- 定义读写锁:使用
PTHREAD_RWLOCK_INITIALIZER
宏初始化读写锁。 - 子线程函数:子线程尝试获取读锁,然后读取共享资源
num
,模拟读取操作,然后释放读锁。 - 创建线程:在
main
函数中创建三个线程,每个线程都执行func
函数。 - 等待线程完成:使用
pthread_join
等待所有线程完成。 - 销毁读写锁:在所有线程完成后,使用
pthread_rwlock_destroy
销毁读写锁。
注意事项:
- 读写锁适用于读多写少的场景,可以提高并发性能。
- 当一个线程持有写锁时,其他线程的读锁和写锁请求都将被阻塞。
- 在使用读写锁时,应该确保锁的初始化和销毁在适当的时候进行,避免资源泄露。
- 读写锁的粒度比互斥锁更细,可以提供更好的并发性,但同时也增加了编程的复杂性。
# 条件变量
在多线程环境中,共享资源的同步是一个常见问题。虽然互斥锁理论上可以解决所有同步问题,但在实际应用中,如果线程的执行依赖于共享资源的动态数值,单纯使用互斥锁可能会导致效率问题。线程可能需要不断地尝试获取锁,检查条件是否满足,然后释放锁,这种行为会导致 CPU 资源的浪费和频繁的上下文切换。
为了解决这种依赖于共享资源条件的同步问题,条件变量提供了一种无竞争的解决方案。它们允许线程在条件不满足时挂起,而不是持续占用 CPU 资源。当条件满足时,线程会被唤醒并继续执行。这种方式减少了不必要的 CPU 占用和上下文切换,提高了程序的效率和响应性。
使用条件变量时,通常需要与互斥锁结合使用,以保护共享资源的访问。线程在进入临界区之前获取互斥锁,然后检查条件是否满足。如果不满足,线程可以释放互斥锁并等待条件变量。当其他线程改变了条件并通知条件变量时,等待的线程会被唤醒,重新获取互斥锁,并再次检查条件。如果条件满足,线程可以继续执行;如果不满足,线程可以再次等待或退出。
# 条件变量的基本原理
条件变量提供了一种机制,允许线程在共享资源的某个条件不满足时挂起。这种挂起是等待状态,直到另一个线程发出通知,告知条件已经满足,从而唤醒等待的线程。
示例场景 :
例如,线程 A 在持有互斥锁的情况下,发现执行所需的条件尚未成熟。在这种情况下,线程 A 可以选择主动进入阻塞状态,并释放持有的锁。这一过程是原子操作,即挂起和解锁是不可分割的步骤。
一旦锁被释放,其他线程,比如线程 B,有机会获取该锁并进行必要的操作。在某些情况下,线程 B 可能不需要持有锁就可以修改条件。
在线程 B 执行其逻辑操作的过程中,如果它判断当前是合适的时机来唤醒线程 A,它可以执行唤醒操作。这个操作会通知等待中的线程 A。
线程 A 在接收到唤醒通知后,会首先恢复运行,并尝试重新获取之前释放的互斥锁。一旦重新获得锁,线程 A 将继续执行它的后续指令。
这个过程中涉及到两个关键动作:一是线程 A 的主动阻塞,二是线程 B 的唤醒操作。
通过条件变量,多线程程序可以更加高效地处理线程间的协作问题,避免了不必要的 CPU 资源浪费,并减少了线程间的冲突。
# pthread_cond_init
条件变量的定义与初始化
在多线程编程中,条件变量是一种同步机制,允许线程在不满足特定条件时挂起。首先,需要定义一个条件变量,并使用 pthread_cond_init
函数进行初始化。此函数接受条件变量的指针和一个可选的条件变量属性对象,通常设置为 NULL。
pthread_cond_init
是 POSIX 线程库中用于初始化条件变量的函数。条件变量是一种同步机制,它允许线程在某些条件不满足时挂起(等待),直到其他线程发出信号表明条件已经满足。
函数原型如下:
#include <pthread.h> | |
int pthread_cond_init(pthread_cond_t *cond, const pthread_condattr_t *attr); |
参数说明:
- pthread_cond_t *cond: 这是一个指向
pthread_cond_t
类型的指针,用于存储新创建的条件变量的标识符。成功初始化后,这个标识符将被用来在后续操作中引用这个条件变量。 - const pthread_condattr_t *attr: 这是一个指向
pthread_condattr_t
类型的指针,它指定了条件变量的属性。如果这个参数是NULL
,条件变量将被初始化为默认属性。
返回值:
- 如果
pthread_cond_init
调用成功,返回 0。 - 如果调用失败,返回一个错误码。例如,如果内存分配失败,将返回
ENOMEM
。
使用场景:
- 当需要在多线程程序中同步线程,基于某些条件进行挂起和唤醒操作时。
- 当一个或多个线程需要等待某个事件的发生,而其他线程负责在适当的时候发出信号。
工作机制:
- 通过调用
pthread_cond_init
,操作系统会分配必要的资源来创建一个条件变量,并根据提供的属性(如果有)初始化它。 - 初始化后,条件变量可以用来与互斥锁配合使用,以实现线程间的协调。
# pthread_cond_wait
线程的阻塞与等待
当线程需要等待某个条件成立时,可以使用 pthread_cond_wait
函数主动进入阻塞状态。此函数会释放与之关联的互斥锁,并使线程进入等待状态,直到被其他线程唤醒。
pthread_cond_wait
是 POSIX 线程库中用于等待(挂起)一个条件变量的函数。当线程执行到这个函数时,它会释放与之配合使用的互斥锁,并进入等待状态,直到另一个线程发出信号或广播唤醒它。
函数原型如下:
#include <pthread.h> | |
int pthread_cond_wait(pthread_cond_t *cond, pthread_mutex_t *mut); |
参数说明:
- pthread_cond_t *cond: 这是一个指向
pthread_cond_t
类型的指针,它标识了等待的条件变量。这个条件变量必须已经通过pthread_cond_init
函数初始化。 - pthread_mutex_t *mut: 这是一个指向
pthread_mutex_t
类型的指针,它标识了与条件变量配合使用的互斥锁。互斥锁必须已经通过pthread_mutex_init
函数初始化,并且在调用pthread_cond_wait
之前被锁定。
返回值:
- 如果条件变量被信号唤醒并且调用成功,返回 0。
- 如果调用失败,返回一个错误码。例如:
EINVAL
: 传入的条件变量或互斥锁参数无效或未初始化。EINTR
: 等待被中断,例如,由于一个信号的传递。
使用场景:
- 当线程需要等待某个条件变为真,而这个条件目前还不满足时。
- 当需要同步线程之间的操作,确保它们在正确的顺序执行。
工作机制:
- 调用
pthread_cond_wait
时,线程首先会释放它所持有的互斥锁,然后进入等待状态。 - 此时,线程会被挂起,不会消耗 CPU 资源。
- 当其他线程调用
pthread_cond_signal
或pthread_cond_broadcast
并释放了至少一个等待该条件变量的线程时,等待的线程会被唤醒。 - 被唤醒的线程将重新尝试获取互斥锁,并在成功获取互斥锁后继续执行。
# pthread_cond_signal
唤醒等待的线程
一旦条件成立,其他线程可以使用 pthread_cond_signal
函数来唤醒等待条件变量的线程。此函数会将一个等待的线程从阻塞状态移至就绪状态,使其能够重新获取锁并继续执行。
pthread_cond_signal
是 POSIX 线程库中用于唤醒等待某个条件变量的一个或多个线程的函数。当使用条件变量进行线程同步时,这个函数允许一个线程向其他线程发出信号,表明它们所等待的条件已经满足。
函数原型如下:
#include <pthread.h> | |
int pthread_cond_signal(pthread_cond_t *cond); |
参数说明:
- pthread_cond_t *cond: 这是一个指向
pthread_cond_t
类型的指针,它标识了要触发的条件变量。这个条件变量必须已经通过pthread_cond_init
函数初始化。
返回值:
- 如果条件变量成功触发,返回 0。
- 如果调用失败,返回一个错误码。例如:
EINVAL
: 传入的条件变量参数无效或未初始化。
使用场景:
- 当一个线程完成了某些工作,并且这项工作是其他线程继续执行所依赖的条件时。
- 当需要唤醒等待条件变量的线程,以便它们可以检查条件并继续执行。
工作机制:
- 当一个线程调用
pthread_cond_signal
时,如果有一个或多个线程正在等待指定的条件变量,操作系统会选择其中一个线程(通常是 FIFO 顺序,但具体取决于实现),使其从pthread_cond_wait
或pthread_cond_timedwait
调用中唤醒。 - 被唤醒的线程将重新尝试获取之前与之配合使用的互斥锁,并在成功获取互斥锁后继续执行。
值得注意的是, pthread_cond_signal
一次只能唤醒一个等待的线程。如果该线程无法立即获取锁,它将再次进入阻塞状态,直到锁被释放。
# pthread_cond_destroy
条件变量的销毁
在条件变量不再需要时,应使用 pthread_cond_destroy
函数将其销毁。这有助于释放与条件变量关联的资源。
pthread_cond_destroy
是 POSIX 线程库中用于销毁条件变量的函数。当条件变量不再被需要时,例如在程序结束或某个模块卸载时,应当使用此函数来释放与条件变量关联的资源。
函数原型如下:
#include <pthread.h> | |
int pthread_cond_destroy(pthread_cond_t *cond); |
参数说明:
- pthread_cond_t *cond: 这是一个指向
pthread_cond_t
类型的指针,它标识了要销毁的条件变量。这个条件变量必须已经通过pthread_cond_init
函数初始化。
返回值:
- 如果
pthread_cond_destroy
调用成功,返回 0。 - 如果调用失败,返回一个错误码。例如:
EINVAL
: 传入的条件变量参数无效或未正确初始化。EBUSY
: 条件变量仍然有线程在等待,不能被销毁。
使用场景:
- 在条件变量生命周期结束时,需要确保释放与之相关的系统资源。
- 在动态创建条件变量的场合,如程序模块初始化时创建,在模块卸载时应当销毁。
工作机制:
- 当调用
pthread_cond_destroy
时,操作系统会检查条件变量的状态。如果条件变量处于无线程等待的安全状态,它将被销毁,相关资源将被释放。 - 如果有线程正在等待该条件变量,销毁操作将失败,并返回
EBUSY
错误。
# 卖票示例
以一个卖票的逻辑为例:
一个人卖票:票未必每一次都能卖掉,每一次买票的人在随机的状态下选择是否买票 。
另一个人加票:在初始 20 张票的情况下,每次卖一张,当第一次票小于 5 张的时候再追加 10 张票。
typedef struct share_state { | |
int ticketNum; | |
int flag; // 0 未加票,1 已经加票 | |
pthread_mutex_t mLock; | |
pthread_cond_t cond; | |
} share_state_t; | |
void *sellFun(void *arg) { | |
share_state_t *pShareState = (share_state_t *)arg; | |
while (1) { | |
pthread_mutex_lock(&pShareState->mLock); | |
if (pShareState->ticketNum <= 0 && pShareState->flag != 0) { | |
pthread_mutex_unlock(&pShareState->mLock); | |
break; | |
} | |
struct timeval nowTime; | |
gettimeofday(&nowTime, NULL); | |
srand((unsigned int)nowTime.tv_usec); | |
double rand_num = (double)rand() / RAND_MAX; | |
if (pShareState->ticketNum > 0 && rand_num < 0.1) { | |
pShareState->ticketNum--; | |
printf("ticketNum = %d \n", pShareState->ticketNum); | |
} | |
if (pShareState->ticketNum <= 5 && pShareState->flag == 0) { | |
pthread_cond_signal(&pShareState->cond); | |
pthread_cond_wait(&pShareState->cond, &pShareState->mLock); | |
} | |
pthread_mutex_unlock(&pShareState->mLock); | |
} | |
return NULL; | |
} | |
void *purchaseFun(void *arg) { | |
share_state_t *pShareState = (share_state_t *)arg; | |
pthread_mutex_lock(&pShareState->mLock); | |
if (pShareState->ticketNum > 5) { | |
pthread_cond_wait(&pShareState->cond, &pShareState->mLock); | |
pShareState->ticketNum = pShareState->ticketNum + 10; | |
pShareState->flag = 1; | |
} else { | |
pShareState->ticketNum = pShareState->ticketNum + 10; | |
pShareState->flag = 1; | |
} | |
pthread_cond_signal(&pShareState->cond); | |
pthread_mutex_unlock(&pShareState->mLock); | |
return NULL; | |
} | |
int main(int argc, char *argv[]) { | |
share_state_t shareState; | |
shareState.ticketNum = 20; | |
shareState.flag = 0; | |
pthread_mutex_init(&shareState.mLock, NULL); | |
pthread_cond_init(&shareState.cond, NULL); | |
pthread_t pid1, pid2; | |
pthread_create(&pid2, NULL, purchaseFun, &shareState); | |
pthread_create(&pid1, NULL, sellFun, &shareState); | |
pthread_join(pid1, NULL); | |
pthread_join(pid2, NULL); | |
return 0; | |
} |
# pthread_cond_timedwait
带有超时限制的条件变量等待
pthread_cond_timedwait
是 POSIX 线程库中的一个函数,它用于实现带有超时限制的条件变量等待。与 pthread_cond_wait
类似, pthread_cond_timedwait
允许线程在等待某个条件变量时挂起,直到被另一个线程通过 pthread_cond_signal
或 pthread_cond_broadcast
唤醒,或者直到超时时间到达。
函数原型如下:
#include <pthread.h> | |
int pthread_cond_timedwait(pthread_cond_t *cond, | |
pthread_mutex_t *mut, | |
const struct timespec *abstime); |
参数说明:
- pthread_cond_t *cond: 这是一个指向
pthread_cond_t
类型的指针,标识了要等待的条件变量。 - pthread_mutex_t *mut: 这是一个指向
pthread_mutex_t
类型的指针,标识了与条件变量配合使用的互斥锁。 - const struct timespec *abstime: 这是一个指向
timespec
结构的指针,timespec
结构指定了超时时间。这个时间是绝对的,表示从某个固定的时间点(如历元或系统启动时间)起的时长。
返回值:
- 如果条件变量被信号唤醒并且调用成功,返回 0。
- 如果超时时间到达并且没有收到信号,返回
ETIMEDOUT
。 - 如果调用失败,返回其他错误码。例如:
EINVAL
: 传入的条件变量或互斥锁参数无效或未初始化。EINTR
: 等待被中断,例如,由于一个信号的传递。
使用场景:
- 当线程需要等待某个条件变量,但不想无限期地等待时。
- 当需要在某个特定时间点放弃等待并继续执行其他任务时。
工作机制:
- 调用
pthread_cond_timedwait
时,线程首先会释放它所持有的互斥锁。 - 然后,线程进入等待状态,直到以下任一情况发生:
- 接收到条件变量的信号或广播,并被唤醒。
- 超时时间到达,仍未收到信号。
- 如果线程被唤醒,它将重新尝试获取互斥锁,并在成功获取互斥锁后继续执行。
示例:
int main(int argc,char*argv[]) | |
{ | |
pthread_mutex_t mLock; | |
pthread_cond_t cond; | |
pthread_mutex_init(&mLock, NULL); | |
pthread_cond_init(&cond, NULL); | |
pthread_mutex_lock(&mLock); | |
time_t now = time(NULL); | |
struct timespec end; | |
end.tv_sec = now+10; | |
end.tv_nsec = 0; | |
int res = pthread_cond_timedwait(&cond, &mLock, &end); | |
THREAD_ERROR_CHECK(res, "timedwait"); | |
pthread_mutex_unlock(&mLock); | |
return 0; | |
} |
# pthread_cond_broadcast
唤醒所有等待指定条件变量的线程
pthread_cond_broadcast
是 POSIX 线程库中的一个函数,它用于唤醒所有等待指定条件变量的线程。与 pthread_cond_signal
函数不同,后者只唤醒一个等待的线程(如果有多个线程在等待的话),而 pthread_cond_broadcast
会唤醒所有等待该条件变量的线程。
函数原型如下:
#include <pthread.h> | |
int pthread_cond_broadcast(pthread_cond_t *cond); |
参数说明:
- pthread_cond_t *cond: 这是一个指向
pthread_cond_t
类型的指针,标识了要广播的条件变量。这个条件变量必须已经通过pthread_cond_init
函数初始化。
返回值:
- 如果所有等待的线程都被成功唤醒,返回 0。
- 如果调用失败,返回一个错误码。例如:
EINVAL
: 传入的条件变量参数无效或未初始化。
使用场景:
- 当有多个线程都在等待同一个条件变量,并且需要同时唤醒它们以进行某些操作时。
- 当条件变量满足时,所有等待的线程都需要立即检查条件并继续执行。
工作机制:
- 当调用
pthread_cond_broadcast
时,所有因pthread_cond_wait
或pthread_cond_timedwait
调用而等待该条件变量的线程都会被唤醒。 - 被唤醒的线程将重新尝试获取之前与之配合使用的互斥锁,并在成功获取互斥锁后继续执行。
typedef struct share_value{ | |
int num; | |
pthread_mutex_t mLock; | |
pthread_cond_t cond; | |
} share_value_t; | |
void *fun(void *arg){ | |
share_value_t *pShareValue = (share_value_t *)arg; | |
pthread_mutex_lock(&pShareValue->mLock); | |
pShareValue->num++; | |
int childNum = pShareValue->num; | |
pthread_mutex_unlock(&pShareValue->mLock); | |
sleep(childNum); | |
printf("i am %d child thread \n", childNum); | |
pthread_mutex_lock(&pShareValue->mLock); | |
printf("i am %d child thread before \n", childNum); | |
pthread_cond_wait(&pShareValue->cond,&pShareValue->mLock); | |
printf("i am %d child thread after \n", childNum); | |
pthread_mutex_unlock(&pShareValue->mLock); | |
} | |
int main(int argc,char*argv[]) | |
{ | |
share_value_t shareValue; | |
shareValue.num = 0; | |
pthread_mutex_init(&shareValue.mLock, NULL); | |
pthread_cond_init(&shareValue.cond, NULL); | |
pthread_t pid1, pid2; | |
pthread_create(&pid1, NULL, fun, &shareValue); | |
pthread_create(&pid2, NULL, fun, &shareValue); | |
sleep(5); | |
pthread_mutex_lock(&shareValue.mLock); | |
pthread_cond_broadcast(&shareValue.cond); | |
// pthread_cond_signal(&shareValue.cond); | |
pthread_mutex_unlock(&shareValue.mLock); | |
pthread_join(pid1, NULL); | |
pthread_join(pid2, NULL); | |
pthread_mutex_destroy(&shareValue.mLock); | |
pthread_cond_destroy(&shareValue.cond); | |
return 0; | |
} |
# 线程属性
当创建线程时,可以通过 pthread_attr_t
类型的变量来指定线程的多种属性,这些属性包括但不限于线程的调度情况、CPU 绑定、内存分配、栈大小和线程的分离状态。以
- 设置线程的 CPU 亲和性: 使用
pthread_attr_setaffinity_np
可以设置线程的 CPU 亲和性,这允许线程只在特定的 CPU 或 CPU 集合上运行。 - 设置线程的脱离状态: 通过
pthread_attr_setdetachstate
,可以设置线程的脱离状态,决定线程是可加入的还是分离的。 - 设置线程的保护区域大小:
pthread_attr_setguardsize
允许设置线程栈的保护区域大小,这有助于防止栈溢出。 - 设置线程的调度属性继承方式: 使用
pthread_attr_setinheritsched
可以设置线程创建时是否继承其父线程的调度属性。 - 设置线程的调度参数:
pthread_attr_setschedparam
用于设置线程的调度参数,包括优先级。 - 设置线程的调度策略:
pthread_attr_setschedpolicy
允许设置线程的调度策略,如先进先出(SCHED_FIFO)、轮转法(SCHED_RR)或标准(SCHED_OTHER)。 - 设置线程的优先级范围: 通过
pthread_attr_setscope
可以设置线程优先级的有效范围,影响线程的调度。 - 设置线程栈的地址和大小: 使用
pthread_attr_setstack
可以同时设置线程栈的地址和大小。 - 设置线程的堆栈地址:
pthread_attr_setstackaddr
允许指定线程栈的内存地址,确保线程栈在特定区域创建。 - 设置线程的堆栈大小:
pthread_attr_setstacksize
用于设置线程栈的大小。
这些属性的设置可以通过相应的 pthread_attr_set*
函数完成,并将设置好的属性对象传递给 pthread_create
函数,以控制新创建线程的行为。
示例:设置线程的分离状态属性
分离状态属性决定了线程终止后是否可以被其他线程通过 pthread_join
捕获其终止状态。如果线程被设置为分离(detached)状态,它将在终止时自动释放其资源,而其他线程调用 pthread_join
将返回错误。
#include <pthread.h> | |
#include <stdio.h> | |
#include <stdlib.h> | |
#define THREAD_ERROR_CHECK(res, msg) \ | |
if (res != 0) { \ | |
fprintf(stderr, "%s failed with error %d\n", msg, res); \ | |
exit(EXIT_FAILURE); \ | |
} | |
void *func(void *arg) { | |
// 线程函数的实现 | |
return NULL; | |
} | |
int main(int argc, char* argv[]) { | |
pthread_attr_t attr; | |
pthread_t tid; | |
// 初始化线程属性对象 | |
pthread_attr_init(&attr); | |
// 设置线程为分离状态 | |
pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED); | |
// 创建线程 | |
pthread_create(&tid, &attr, func, NULL); | |
// 等待线程终止,由于线程是分离状态,这里将返回错误 | |
int res = pthread_join(tid, NULL); | |
THREAD_ERROR_CHECK(res, "join"); | |
// 销毁线程属性对象 | |
pthread_attr_destroy(&attr); | |
return 0; | |
} |
# 线程安全问题
多线程共享同一进程的地址空间,因此在访问共享数据时可能会遇到竞争条件。这种问题不仅可能发生在自定义函数中,也可能出现在库函数的使用中。例如, ctime
函数会将转换后的日期时间字符串存储在静态缓冲区中,这可能导致线程间的不安全行为。
* 以下是使用 ctime
函数的示例代码,它展示了线程安全问题:
#include <stdio.h> | |
#include <time.h> | |
#include <pthread.h> | |
void *func(void *arg) { | |
time_t childTime; | |
time(&childTime); | |
char *str = ctime(&childTime); | |
printf("child time = %s\n", str); | |
sleep(5); | |
printf("child time = %s\n", str); // 这里可能会看到与第一次不同的输出 | |
return NULL; | |
} | |
int main(int argc, char* argv[]) { | |
pthread_t tid; | |
pthread_create(&tid, NULL, func, NULL); | |
sleep(3); | |
time_t mainTime; | |
time(&mainTime); | |
printf("main time = %s\n", ctime(&mainTime)); | |
pthread_join(tid, NULL); | |
return 0; | |
} |
为了解决这个问题,可以使用 ctime_r
函数,它允许指定一个用户定义的缓冲区来存储结果,从而避免了线程间的共享数据问题。
#include <stdio.h> | |
#include <time.h> | |
#include <pthread.h> | |
void *func(void *arg) { | |
time_t childTime; | |
time(&childTime); | |
char buff[100]; | |
char *str = ctime_r(&childTime, buff); | |
printf("child time = %s\n", str); | |
sleep(5); | |
printf("child time = %s\n", str); // 使用缓冲区,输出将是一致的 | |
return NULL; | |
} | |
int main(int argc, char* argv[]) { | |
pthread_t tid; | |
pthread_create(&tid, NULL, func, NULL); | |
sleep(3); | |
time_t mainTime; | |
time(&mainTime); | |
char buff[100]; | |
printf("main time = %s\n", ctime_r(&mainTime, buff)); | |
pthread_join(tid, NULL); | |
return 0; | |
} |
在帮助手册或文档中,库函数的作者通常会说明其线程安全属性。开发者应当注意这些属性,并在必要时选择线程安全的替代函数。
# 可重入性
在多线程编程中,可重入性(reentrancy)是一个重要的概念。一个函数是可重入的,如果它在被一个线程执行的过程中,可以再次被同一个线程或另一个线程调用,而不会导致任何错误或不一致的问题。可重入函数通常不依赖于全局或静态数据,或者在访问这些数据时使用适当的同步机制。
下面示例代码中, fun
函数是不可重入的,因为它使用了一个静态数组 p
来存储结果。如果 fun
函数在第一次调用时被中断,并且再次被调用,它可能会返回一个未完成的字符串,或者覆盖之前的结果。这是因为静态数组 p
在函数调用之间保持其状态,而没有适当的同步机制来保护对它的访问。
#include <testfun.h> | |
char *fun(){ | |
static char p[20] = {0}; | |
for(int i=0; i<20; i++){ | |
if(p[i] == 0){ | |
p[i] = 'a'; | |
break; | |
} | |
} | |
return p; | |
} | |
int main(int argc, char* argv[]){ | |
char *p1 = fun(); | |
printf("p = %s \n", p1); | |
char *p2 = fun(); | |
printf("p = %s \n", p2); | |
return 0; | |
} |
要使 fun
函数可重入,可以采取以下措施:
- 移除静态存储类,使用局部变量或传入的参数来代替。
- 如果必须使用静态或全局数据,确保使用互斥锁或其他同步机制来保护对它们的访问。
实现可重入函数的关键点包括:
- 避免使用静态或全局变量,除非可以通过加锁来保护它们。
- 不要调用非可重入的库函数,除非在调用它们时使用适当的同步。
- 避免在函数中分配或释放内存,这通常涉及到非可重入的函数,如
malloc
和free
。 - 确保函数在被中断时能够安全地恢复执行,不会导致数据损坏或不一致。
关于 malloc
函数的不可重入性。 malloc
可能会修改其内部状态,如内存池或分配的内存块计数器。在多线程环境中,如果一个线程在 malloc
被调用后和完成前被阻塞,而另一个线程尝试调用 malloc
,可能会导致死锁或不一致的问题。为了安全地在多线程程序中使用动态内存分配,可以使用线程安全的内存分配函数,或者在分配内存时使用互斥锁来同步访问。
在信号处理中,如果信号处理函数调用了不可重入的函数,如 malloc
,并且此时发生了线程切换,切换到的线程也调用了 malloc
,就可能发生死锁。为了避免这种情况,应该确保信号处理函数尽可能简洁,只执行必要的操作,并且避免调用不可重入的函数。如果需要在信号处理函数中进行复杂的操作,可以考虑使用 sigsetjmp
和 siglongjmp
来保存和恢复程序的状态,或者使用其他同步机制来确保安全。