Linux 线程同步

一、线程同步的概念

  

  在人们的日常生活中的锁大概有两种:一种是不允许访问;另一种是资源忙,同一时间只允许一个使用者占用,其它使用者必须要等待。

  1)不允许访问的锁容易理解,就像每家每户的门锁,不允许外人进入。

  2)第二种锁,例如火车上的厕所,它是公共的,空闲的时候任何人可以进入,人进去以后就会把它锁起来,其它的人如果要上厕所,必须等待解锁,即里面的人出来。还有红绿灯,红灯是加锁,绿灯是解锁。

  对多线程来说,资源是共享的,基本上不存在不允许访问的情况,但是,共享的资源在某一时间点只能有一个线程占用,所以需要给资源加锁。

  程同步j就是锁共享资源,线程给共享资源加的锁。

  线程的锁的种类有互斥锁、读写锁、条件变量、自旋锁、信号灯。

  在本章节中,只介绍互斥锁。

二、互斥锁

  互斥锁机制是同一时刻只允许一个线程占有共享的资源。

1、初始化锁

int pthread_mutex_init(pthread_mutex_t *mutex,const pthread_mutex_attr_t *mutexattr);

  其中参数 mutexattr 用于指定锁的属性(见下),如果为NULL则使用缺省属性。

  互斥锁的属性在创建锁的时候指定,当资源被某线程锁住的时候,其它的线程在试图加锁时表现将不同。当前有四个值可供选择:

  1)PTHREAD_MUTEX_TIMED_NP,这是缺省值,也就是普通锁。当一个线程加锁以后,其余请求锁的线程将形成一个等待队列,并在解锁后按优先级获得锁。这种锁策略保证了资源分配的公平性。

  2)PTHREAD_MUTEX_RECURSIVE_NP,嵌套锁,允许同一个线程对同一个锁成功获得多次,并通过多次unlock解锁。

  3)PTHREAD_MUTEX_ERRORCHECK_NP,检错锁,如果同一个线程请求同一个锁,则返回EDEADLK,否则与PTHREAD_MUTEX_TIMED_NP类型动作相同。

  4)PTHREAD_MUTEX_ADAPTIVE_NP,适应锁,动作最简单的锁类型,等待解锁后重新竞争。

2、阻塞加锁

int pthread_mutex_lock(pthread_mutex *mutex);

  如果是锁是空闲状态,本线程将获得这个锁;如果锁已经被占据,本线程将排队等待,直到成功的获取锁。

3、非阻塞加锁

int pthread_mutex_trylock( pthread_mutex_t *mutex);

  该函数语义与 pthread_mutex_lock() 类似,不同的是在锁已经被占据时立即返回EBUSY,不是挂起等待。

  这个就像是日常生活中,如果厕所里面有10个坑位,我从第一个开始看是不是空闲的。如果不是,我就看第二个,以此类推。

4、解锁

int pthread_mutex_unlock(pthread_mutex *mutex);

线程把自己持有的锁释放。

5、销毁锁(此时锁必需unlock状态,否则返回EBUSY)

int pthread_mutex_destroy(pthread_mutex *mutex);

销毁锁之前,锁必需是空闲状态(unlock)。

三、示例程序

  多线程可以共享资源(变量和对象),对编程带来了方便,但是某些对象虽然可以共享,但在同一个时间只能由一个线程使用,多个线程同时使用会产生冲突,例如socket连接,数据库连接池。

我们把前几章节的socket客户端程序book247.cpp修改为多线程。

示例(book263.cpp)


#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <netdb.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <pthread.h>

//xx pthread_mutex_t mutex; // 申明一个互斥锁

// 与客户端通信线程的主函数
void *pth_main(void *arg)
{
  int pno=(long)arg;   // 线程编号

  pthread_detach(pthread_self());

  char strbuffer[1024];
  
  for (int ii=0;ii<3;ii++)    // 与服务端进行3次交互。
  {
    //xx pthread_mutex_lock(&mutex);  // 加锁
    memset(strbuffer,0,sizeof(strbuffer));
    sprintf(strbuffer,"线程%d:这是第%d个超级女生,编号%03d。",pno,ii+1,ii+1);
    if (TcpClient.Send(strbuffer,strlen(strbuffer))<=0) break;
    printf("发送:%s\n",strbuffer);

    memset(strbuffer,0,sizeof(strbuffer));
    if (TcpClient.Recv(strbuffer,sizeof(strbuffer))<=0) break;
    printf("线程%d接收:%s\n",pno,strbuffer);
    //xx pthread_mutex_unlock(&mutex);  // 释放锁
    // usleep(100);   // usleep(100),否则其它的线程无法获得锁。
  }

  pthread_exit(0);
}

int main()
{
  // 向服务器发起连接请求
  if (TcpClient.ConnectToServer("172.16.0.15",5051)==false)
  { printf("TcpClient.ConnectToServer(\"172.16.0.15\",5051) failed,exit...\n"); return -1; }

  //xx pthread_mutex_init(&mutex,0); // 创建锁

  pthread_t pthid1,pthid2;
  pthread_create(&pthid1,NULL,pth_main,(void*)1);   // 创建第一个线程
  pthread_create(&pthid2,NULL,pth_main,(void*)2);   // 创建第二个线程

  pthread_join(pthid1,NULL);    // 等待线程1退出。
  pthread_join(pthid2,NULL);    // 等待线程2退出。

  //xx pthread_mutex_lock(&mutex);   // 销毁锁
}

1.代码解析

  现在学习的是线程同步,也就是给线程中给共享资源加锁,那么在上面的程序中,锁的共享资源是什么?

 for (int ii=0;ii<3;ii++)    // 与服务端进行3次交互。
  {
    pthread_mutex_lock(&mutex);  // 加锁
    memset(strbuffer,0,sizeof(strbuffer));
    sprintf(strbuffer,"线程%d:这是第%d个超级女生,编号%03d。",pno,ii+1,ii+1);
    if (TcpClient.Send(strbuffer,strlen(strbuffer))<=0) break;
    printf("发送:%s\n",strbuffer);

    memset(strbuffer,0,sizeof(strbuffer));
    if (TcpClient.Recv(strbuffer,sizeof(strbuffer))<=0) break;
    printf("线程%d接收:%s\n",pno,strbuffer);
    pthread_mutex_unlock(&mutex);  // 释放锁
    usleep(100);   // usleep(100),否则其它的线程无法获得锁。
  }

  仔细回想一下,客户端与服务端进行通信的共享资源是什么,是用于通信的socket。再想一下之前封装的socket通信客户端类,成员变量就是用于与服务端进行通信的socket。

  或者从上面这段代码思考,它是从什么时候开始加锁的,从什么时候解锁。加锁是从给服务端发报文之前;解锁是发送完了报文,接收到服务端的回复后。收发报文用的正是用于通信的socket。

2.为什么要加锁呢

  为什么要给共享资源加锁呢?就拿这个通信的例子来讲,某天我用村子里唯一一部电话与刘亦菲通话,谈一下那个一个亿的小项目 。张三也想和刘亦菲通话,因为他和刘亦菲谈有一个300万的项目。所以我们规定,我先去和刘亦菲谈。
  只有我和刘亦菲谈项目时,张三在外面等待。我相当于对共享资源(电话)加锁了,现在只有我能使用,这样别人就不能干扰我们。

  但是如果我和张三一起使用电话,一起说话,就会有可能会出错。比如说刘亦菲问我的项目要多少投资,本来我想要一亿,但是张三也在说话,他就说要一百万。刘亦菲以为是我要一百万,就给了我一百万。那这样不就出错了吗。

  简单点说就是,如果你和你女朋友正在腻歪,如果通话没有加锁。突然有人就插了进来,说了一句:“我要打死你”。然后这时候你又能和你女朋友通话了,可能你就吃不了兜着走了。加锁防止别的线程干扰,造成收发信息顺序错误。下面就来测试一下。

3.测试加锁与不加锁

  在book263.cpp程序中,客户端成功连上服务器后,创建两个线程,同时与服务端进行通信,发送3个请求报文并接收服务端的回应。

book263.cpp暂时不启用锁,先试试效果。

启动服务端程序book261,然后再启动book263。

运行效果

在这里插入图片描述

  大家仔细研究一下book263运行的结果,可以发现客户端的两个线程的报文收发出现了混乱。

把book263.cpp的线程锁代码启用,编译运行。

运行效果

在这里插入图片描述
非常棒,这正在我们想要的结果。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值