进程是Linux资源分配的对象,Linux会为进程分配虚拟内存(4G)和文件句柄等资源,是一个静态的概念。线程是CPU调度的对象,是一个动态的概念。一个进程之中至少包含有一个或者多个线程。这些线程共享该进程空间的内存和文件句柄资源,多个线程竞争地获得这些资源。为了防止多个线程访问资源的不一致性,多线程编程一个很重要的任务就是控制好线程同步。本文简单介绍一下Linux的同步对象和使用时的一些注意事项。
1、互斥量(Mutex)
互斥量本质上讲是一把锁,该锁保护一个或者一些资源(内存或者文件句柄等数据)。一个线程如果需要访问该资源必须要获得互斥量,并对其加锁。这时如果其他线程如果想访问该资源也必须要获得该互斥量,但是锁已经加锁,所以这些进程只能阻塞,直到获得该锁的线程解锁。这时阻塞的线程里面有一个线程获得该互斥量并加锁,获准访问该资源。其他的线程继续阻塞,周而复始。
Linux互斥量句柄为pthread_mutex_t。可以以PTHREAD_MUTEX_INITIALIZER初始化一个互斥量,或者调用如下函数动态进行初始化:
#include <pthread.h>
int pthread_mutex_init(pthread_mutex_t *restrict mutex,
const pthread_mutexattr_t *restrict attr);
销毁一个互斥量调用如下函数:
#include <pthread.h>
int pthread_mutex_destroy(pthread_mutex_t *mutex);
对一个互斥量加锁和解锁函数如下:
#include <pthread.h>
int pthread_mutex_lock(pthread_mutex_t *mutex);
int pthread_mutex_trylock(pthread_mutex_t *mutex);
int pthread_mutex_unlock(pthread_mutex_t *mutex);
当前线程调用pthread_mutex_lock函数时,如果该互斥量未加锁,则当前线程获得该互斥量并解锁,该函数返回;如果当前该互斥量已经加锁,则该函数将会阻塞,直到该互斥量解锁,当前线程获得该互斥量,并加锁返回。
pthread_mutex_trylock如果互斥量为未加锁,则当前线程将会获得该互斥量并加锁。当互斥量为加锁状态,该函数将会立即返回错误EBUSY,不会阻塞当前线程。
互斥量的解锁函数为pthread_mutex_unlock,这样将会释放互斥量资源。
另外注意一个问题就是互斥量死锁(dead lock)的问题。当一个互斥量的时候,不会发生互斥量的问题。当有多个互斥量的时候,有可能发生死锁。例如:有互斥量A,B。假如第一个线程获得互斥量A,并加锁,这时他尝试获得互斥量B,但是互斥量B已经加锁,该线程被阻塞,等待互斥量B。同时另外一个线程先获得互斥量B,并已加锁。这时尝试获得互斥量A,发现互斥量A已经加锁,则阻塞该线程,等待互斥量A。这样出现两个线程互相等待对方已经获得的信号量的问题,都处于阻塞状态,出现死锁。那么怎样解决这种死锁问题呢?那就是线程以同样的顺序获得互斥量。第一个线程先获得互斥量A,再获得互斥量B;第二个线程也以同样的顺序获得互斥量。这样就不会出现死锁的状态了。
但是在一些结构复杂的程序中,很难保证以同样的顺序获得互斥量,那么怎样解决死锁问题呢?就是以pthread_mutex_trylock来尝试获得互斥量,如果不能获得互斥量,则释放已经持有的互斥量。过段时间,再次进行同样的尝试,这样可以避免死锁。
分享到:
相关推荐
本资源提供的"linux下线程池源代码"实现了这样一个线程池模型,具有良好的封装性和易用性,使用者只需两步——创建线程池和添加任务——即可方便地利用线程池执行多任务。 首先,线程池的核心概念是“池化”,即...
Linux下线程和锁的实验代码 代码演示了在LINUX下创建线程并互斥的访问临界资源
VC下线程同步的三种方法(互斥、事件、临界区)简单的代码,让你更容易理解三种同步方式。
在深入探讨Linux下线程池的C语言实现之前,我们首先需要理解线程池的基本概念以及它在系统设计中的重要性。线程池是一种管理线程的机制,它预先创建一组固定数量的线程,等待任务的到来,从而避免了频繁创建和销毁...
本篇将详细介绍在Linux环境下,如何使用C++进行线程的创建、同步以及销毁等基本操作。 一、线程创建 在C++11及以后的版本中,C++标准库提供了一个名为`std::thread`的类来支持线程操作。创建一个新线程可以使用`std...
4. **线程同步**:如果多个线程可能同时操作同一个资源,如共享数据结构或文件,需要使用互斥锁、读写锁或条件变量来确保同步。 5. **处理客户端请求**:在线程中读取客户端数据,进行业务处理,然后向客户端发送...
4. **同步机制**:如互斥量(`std::mutex`)、条件变量(`std::condition_variable`)等,用于保证线程间的正确同步,防止数据竞争。 在实际使用时,用户首先需要定义自己的任务类型,这个类型需要有一个无参构造...
singleton是最常见的设计模式,但是要设计好却是不容易,尤其是多线程的时候,需要考虑线程安全的问题.
在Linux环境下,线程安全的Singleton模板是一种至关重要的设计模式,尤其在多线程编程中。Singleton模式的主要目的是确保一个类只有一个实例,并提供一个全局访问点。这种模式在系统资源管理、缓存、日志记录等方面...
首先,我们需要定义一个结构体来存储线程池的相关信息,包括线程数组、任务队列、同步机制(如互斥锁和条件变量)等。以下是一个简单的线程池结构体示例: ```c typedef struct { pthread_t *threads; // 存储线程...
线程锁,或互斥量,是一种同步对象,它允许同一时间只有一个线程访问被保护的资源。当一个线程获得锁后,其他试图获取该锁的线程将被阻塞,直到拥有锁的线程释放它。这样就确保了在任意时刻,对临界区的访问是互斥的...
在Linux下,可以使用互斥量(mutex)和条件变量(condition variable)来实现线程安全的数据结构,确保多线程环境下的正确同步。 2. **工作队列**:用于存储待执行的任务。线程池管理器将新任务放入队列,而工作...
这通常涉及到同步原语,如互斥锁(mutex)或条件变量(condition variable)的使用。 【参数处理】在设计线程池时,需要考虑如何处理各种输入参数,如线程池大小、任务队列长度等。合理的参数设置直接影响线程池的...
线程互斥是一种同步机制,确保同一时间只有一个线程可以访问共享资源,以避免数据竞争和不一致的状态。在Windows API中,我们可以通过创建和操作Mutex对象来实现这一功能。Mutex有两个主要的API函数:`CreateMutex()...
本文将详细介绍Linux下线程池的实现原理、设计模式以及如何进行线程同步。 线程池的基本概念: 1. **工作线程(Worker Thread)**:线程池中的线程,负责执行提交到线程池的任务。 2. **任务队列(Task Queue)**:...
5. **线程同步**:为了保证线程安全,线程池的实现需要考虑线程间的同步问题,比如对任务队列的访问,可能需要使用互斥锁(mutex)、条件变量(condition variable)等同步原语。 6. **线程管理和回收**:当线程...
Linux下编写一个内核模块,分别线性遍历所有进程和DFS遍历进程树
MFC提供了一系列的同步对象,如`CSemaphore`(信号量)、`CCriticalSection`(临界区)、`CMutex`(互斥量)和`CEvent`(事件)。这些对象用于控制线程间的访问权限和同步,防止资源竞争,确保数据一致性。 4. **...