Linux多线程编程-互斥锁

互斥锁

        多线程编程中,(多线程编程)可以用互斥锁(也称互斥量)可以用来保护关键代码段,以确保其独占式的访问,这有点像二进制信号量。POSIX互斥锁相关函数主要有以下5个:

 

 
  1. #include <pthread.h>

  2. int pthread_mutex_init(pthread_mutex_t *mutex, const pthread_mutexattr_t *mutexattr);

  3. int pthread_mutex_destroy(pthread_mutex_t *mutex);

  4. int pthread_mutex_lock(pthread_mutex_t *mutex);

  5. int pthread_mutex_trylock(pthread_mutex_t *mutex);

  6. int pthread_mutex_unlock(pthread_mutex_t *mutex);

        这些函数第一个参数mutex指向要操作的目标互斥锁,成功时返回0,出错返回错误码

l  pthread_mutex_init用于初始化互斥锁,mutexattr用于指定互斥锁的属性,若为NULL,则表示默认属性。除了用这个函数初始化互斥所外,还可以用如下方式初始化:pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;

l  pthread_mutex_destroy用于销毁互斥锁,以释放占用的内核资源,销毁一个已经加锁的互斥锁将导致不可预期的后果

l  pthread_mutex_lock以原子操作给一个互斥锁加锁。如果目标互斥锁已经被加锁,则pthread_mutex_lock则被阻塞,直到该互斥锁占有者把它给解锁

l  pthread_mutex_trylock和pthread_mutex_lock类似,不过它始终立即返回,而不论被操作的互斥锁是否加锁,是pthread_mutex_lock的非阻塞版本。当目标互斥锁未被加锁时,pthread_mutex_trylock进行加锁操作;否则将返回EBUSY错误码。注意:这里讨论的pthread_mutex_lock和pthread_mutex_trylock是针对普通锁而言的,对于其他类型的锁,这两个加锁函数会有不同的行为

l  pthread_mutex_unlock以原子操作方式给一个互斥锁进行解锁操作。如果此时有其他线程正在等待这个互斥锁,则这些线程中的一个将获得它

 

互斥锁示例程序

 

 
  1. #include <stdio.h>

  2. #include <stdlib.h>

  3. #include <pthread.h>

  4.  
  5. #define err_sys(msg) \

  6. do { perror(msg); exit(-1); } while(0)

  7. #define err_exit(msg) \

  8. do { fprintf(stderr, msg); exit(-1); } while(0)

  9.  
  10. int glab = 1;

  11.  
  12. void *r1(void *arg)

  13. {

  14. pthread_mutex_t* mutex = (pthread_mutex_t *)arg;

  15. static int cnt = 10;

  16.  
  17. while(cnt--)

  18. {

  19. pthread_mutex_lock(mutex);

  20. glab++;

  21. printf("I am in r1. cnt = %d\n", glab);

  22. pthread_mutex_unlock(mutex);

  23. sleep(1);

  24. }

  25. return "r1 over";

  26. }

  27.  
  28. void *r2(void *arg)

  29. {

  30. pthread_mutex_t* mutex = (pthread_mutex_t *)arg;

  31. static int cnt = 10;

  32.  
  33. while(cnt--)

  34. {

  35. pthread_mutex_lock(mutex);

  36. glab++;

  37. printf("I am in r2. cnt = %d\n", glab);

  38. pthread_mutex_unlock(mutex);

  39. sleep(1);

  40. }

  41. return "r2 over";

  42. }

  43.  
  44. int main(void)

  45. {

  46. pthread_mutex_t mutex;

  47. pthread_t t1, t2;

  48. char* p1 = NULL;

  49. char* p2 = NULL;

  50.  
  51. if(pthread_mutex_init(&mutex, NULL) < 0)

  52. err_sys("sem_init error");

  53. pthread_create(&t1, NULL, r1, &mutex);

  54. pthread_create(&t2, NULL, r2, &mutex);

  55.  
  56. pthread_join(t1, (void **)&p1);

  57. pthread_join(t2, (void **)&p2);

  58. pthread_mutex_destroy(&mutex);

  59. printf("s1: %s\n", p1);

  60. printf("s2: %s\n", p2);

  61.  
  62. return 0;

  63. }

 

互斥锁属性

 

        pthread_mutexattr_t结构体定义了一套完整的互斥锁属性。线程库提供了一系列函数来操作pthread_mutexattr_t类型变量,以方便我们获取和设置互斥锁属性。一下是一些主要的函数:

 
  1. #include <pthread.h>

  2. int pthread_mutexattr_destroy(pthread_mutexattr_t *attr);

  3. int pthread_mutexattr_init(pthread_mutexattr_t *attr);

  4. int pthread_mutexattr_getpshared(const pthread_mutexattr_t *restrict attr, int *restrict pshared);

  5. int pthread_mutexattr_setpshared(pthread_mutexattr_t *attr, int pshared);

  6. int pthread_mutexattr_gettype(const pthread_mutexattr_t *restrict attr, int *restrict type);

  7. int pthread_mutexattr_settype(pthread_mutexattr_t *attr, int type);

        互斥锁两种常用属性:pshared和type

        互斥锁属性pshared指定是否允许跨进程共享互斥锁,其可选值有两个:

l  PTHREAD_PROCESS_SHARED。互斥锁可以被跨进程共享。

l  PTHREAD_PROCESS_PRIVATE。互斥锁只能被和锁的初始化线程隶属于同一个进程的线程共享。

互斥锁属性type指定互斥锁的类型。Linux支持如下4种类型的互斥锁:

l  PTHREAD_MUTEX_NORMAL,普通锁。这是互斥锁默认的类型。当一个线程对一个普通锁加锁以后,其余请求该所的线程将形成一个等待队列,并在该所解锁后按优先级获得它。这种锁类型保证了资源分配的公平性。但这种锁也很容易引发问题:一个线程如果对一个已经加锁的普通锁再次加锁,将引发死锁;对一个已经被其他线程加锁的普通锁解锁,或者对一个已经解锁的普通锁解锁将导致不可预期的后果。

l  PTHREAD_MUTEX_ERRORCHECK,检错锁。一个线程如果对一个已经加锁的检错锁再次加锁,则加锁操作返回EDEADLK。对一个已经被其让他线程加锁的检错锁解锁,或者对一个已经解锁的检错锁再次解锁,则检错锁返回EPERM。

l  PTHREAM_MUTEX_RECURSIVE,嵌套锁。这种锁允许一个线程在释放锁之前对他加锁而不发生死锁。不过其他线程如果要获得这个锁,则当前锁的拥有者必须执行相应次数的解锁操作。对一个已经被其他线程枷锁的嵌套锁解锁,或者对一个已经解锁的嵌套锁再次解锁,则解锁操作返回EPERM。

l  PTHREAD_MUTEX_DEFAULT,默认锁。一个线程如果对一个已经加锁的默认锁再次加锁,或者对一个已经被其他线程加锁的默认锁解锁,或者对一个已经解锁的默认锁再次解锁,将导致不可预期的后果。

 

使用嵌套锁(PTHREAM_MUTEX_RECURSIVE)程序示例

 

 
  1. #include <stdio.h>

  2. #include <stdlib.h>

  3. #include <pthread.h>

  4.  
  5. int a = 1;

  6. int b = 1;

  7. pthread_mutex_t mutex;

  8.  
  9. void *pthread1(void *arg)

  10. {

  11. while(1)

  12. {

  13. if(pthread_mutex_lock(&mutex) != 0)

  14. {

  15. printf("1 error\n");

  16. continue;

  17. }

  18. if(pthread_mutex_lock(&mutex) != 0) /* 此时调用了2次lock操作 */

  19. {

  20. printf("1 error\n");

  21. continue;

  22. }

  23. a++;

  24. b++;

  25. if(a != b)

  26. printf("pthread1: %d, %d\n", a, b);

  27. else

  28. printf("11\n");

  29. pthread_mutex_unlock(&mutex);

  30. pthread_mutex_unlock(&mutex);

  31. sleep(1);

  32. }

  33. }

  34.  
  35. void *pthread2(void *arg)

  36. {

  37. while(1)

  38. {

  39. if(pthread_mutex_lock(&mutex) != 0)

  40. {

  41. printf("2 error\n");

  42. continue;

  43. }

  44. a++;

  45. b++;

  46. if(a != b)

  47. printf("pthread2: %d, %d\n", a, b);

  48. else

  49. printf("22------------\n");

  50. pthread_mutex_unlock(&mutex);

  51. sleep(1);

  52. }

  53. }

  54.  
  55. int main(void)

  56. {

  57. pthread_t tid1, tid2;

  58. pthread_mutexattr_t attr;

  59.  
  60. pthread_mutexattr_init(&attr);

  61. pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE);

  62. pthread_mutex_init(&mutex, &attr);

  63. pthread_create(&tid1, NULL, pthread1, NULL);

  64. pthread_create(&tid2, NULL, pthread2, NULL);

  65.  
  66. pthread_join(tid1, NULL);

  67. pthread_join(tid2, NULL);

  68. pthread_mutex_destroy(&mutex);

  69.  
  70. return 0;

  71. }

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值