`
gstarwd
  • 浏览: 1525263 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

Linux下c开发 之 线程通信

阅读更多

1.Linux“线程”


进程与线程之间是有区别的,不过Linux内核只提供了轻量进程的支持,未实现线程模型。Linux是一种“多进程单线程”的操作系统。Linux本身只有进程的概念,而其所谓的“线程”本质上在内核里仍然是进程。

大家知道,进程是资源分配的单位,同一进程中的多个线程共享该进程的资源(如作为共享内存的全局变量)。Linux中所谓的“线程”只是在被创建时 clone了父进程的资源,因此clone出来的进程表现为“线程”,这一点一定要弄清楚。因此,Linux“线程”这个概念只有在打冒号的情况下才是最准确的。



目前Linux中最流行的线程机制为LinuxThreads,所采用的就是线程-进程“一对一”模型,调度交给核心,而在用户级实现一个包括信号处理在内的线程管理机制。LinuxThreads由Xavier Leroy (Xavier.Leroy@inria.fr)负责开发完成,并已绑定在GLIBC中发行,它实现了一种BiCapitalized面向Linux的Posix 1003.1c “pthread”标准接口。Linuxthread可以支持Intel、Alpha、MIPS等平台上的多处理器系统。



  按照POSIX 1003.1c 标准编写的程序与Linuxthread 库相链接即可支持Linux平台上的多线程,在程序中需包含头文件pthread. h,在编译链接时使用命令:



gcc -D -REENTRANT -lpthread xxx. c



  其中-REENTRANT宏使得相关库函数(如stdio.h、errno.h中函数) 是可重入的、线程安全的(thread-safe),-lpthread则意味着链接库目录下的libpthread.a或libpthread.so文件。使用Linuxthread库需要2.0以上版本的Linux内核及相应版本的C库(libc 5.2.18、libc 5.4.12、libc 6)。





2.“线程”控制



  线程创建



  进程被创建时,系统会为其创建一个主线程,而要在进程中创建新的线程,则可以调用pthread_create:


pthread_create(pthread_t *thread, const pthread_attr_t *attr, void *

(start_routine)(void*), void *arg);
 





  start_routine为新线程的入口函数,arg为传递给start_routine的参数。



  每个线程都有自己的线程ID,以便在进程内区分。线程ID在pthread_create调用时回返给创建线程的调用者;一个线程也可以在创建后使用pthread_self()调用获取自己的线程ID:


pthread_self (void) ;
 





  线程退出



  线程的退出方式有三:

  (1)执行完成后隐式退出;

  (2)由线程本身显示调用pthread_exit 函数退出;

pthread_exit (void * retval) ;
 



  (3)被其他线程用pthread_cance函数终止:

pthread_cance (pthread_t thread) ;
 





  在某线程中调用此函数,可以终止由参数thread 指定的线程。



  如果一个线程要等待另一个线程的终止,可以使用pthread_join函数,该函数的作用是调用pthread_join的线程将被挂起直到线程ID为参数thread的线程终止:


pthread_join (pthread_t thread, void** threadreturn);
 




3.线程通信



  线程互斥

  互斥意味着“排它”,即两个线程不能同时进入被互斥保护的代码。Linux下可以通过pthread_mutex_t 定义互斥体机制完成多线程的互斥操作,该机制的作用是对某个需要互斥的部分,在进入时先得到互斥体,如果没有得到互斥体,表明互斥部分被其它线程拥有,此时欲获取互斥体的线程阻塞,直到拥有该互斥体的线程完成互斥部分的操作为止。



  下面的代码实现了对共享全局变量x 用互斥体mutex 进行保护的目的:

int x; // 进程中的全局变量

pthread_mutex_t mutex;

pthread_mutex_init(&mutex, NULL); //按缺省的属性初始化互斥体变量mutex

pthread_mutex_lock(&mutex); // 给互斥体变量加锁

… //对变量x 的操作

phtread_mutex_unlock(&mutex); // 给互斥体变量解除锁

 



  线程同步



  同步就是线程等待某个事件的发生。只有当等待的事件发生线程才继续执行,否则线程挂起并放弃处理器。当多个线程协作时,相互作用的任务必须在一定的条件下同步。



  Linux下的C语言编程有多种线程同步机制,最典型的是条件变量(condition variable)。pthread_cond_init用来创建一个条件变量,其函数原型为:


pthread_cond_init (pthread_cond_t *cond, const pthread_condattr_t *attr);

 



  pthread_cond_wait和pthread_cond_timedwait用来等待条件变量被设置,值得注意的是这两个等待调用需要一个已经上锁的互斥体mutex,这是为了防止在真正进入等待状态之前别的线程有可能设置该条件变量而产生竞争。pthread_cond_wait的函数原型为:


pthread_cond_wait (pthread_cond_t *cond, pthread_mutex_t *mutex);



  pthread_cond_broadcast用于设置条件变量,即使得事件发生,这样等待该事件的线程将不再阻塞:



pthread_cond_broadcast (pthread_cond_t *cond) ;



  pthread_cond_signal则用于解除某一个等待线程的阻塞状态:



pthread_cond_signal (pthread_cond_t *cond) ;



  pthread_cond_destroy 则用于释放一个条件变量的资源。

 



  在头文件semaphore.h 中定义的信号量则完成了互斥体和条件变量的封装,按照多线程程序设计中访问控制机制,控制对资源的同步访问,提供程序设计人员更方便的调用接口。

sem_init(sem_t *sem, int pshared, unsigned int val);



  这个函数初始化一个信号量sem 的值为val,参数pshared 是共享属性控制,表明是否在进程间共享。



sem_wait(sem_t *sem);
 




  调用该函数时,若sem为无状态,调用线程阻塞,等待信号量sem值增加(post )成为有信号状态;若sem为有状态,调用线程顺序执行,但信号量的值减一。


sem_post(sem_t *sem);

 



  调用该函数,信号量sem的值增加,可以从无信号状态变为有信号状态。





4.实例



  下面我们还是以名的生产者/消费者问题为例来阐述Linux线程的控制和通信。一组生产者线程与一组消费者线程通过缓冲区发生联系。生产者线程将生产的产品送入缓冲区,消费者线程则从中取出产品。缓冲区有N 个,是一个环形的缓冲池。

#include

#include

#define BUFFER_SIZE 16 // 缓冲区数量

struct prodcons

{

// 缓冲区相关数据结构

int buffer[BUFFER_SIZE]; /* 实际数据存放的数组*/

pthread_mutex_t lock; /* 互斥体lock 用于对缓冲区的互斥操作 */

int readpos, writepos; /* 读写指针*/

pthread_cond_t notempty; /* 缓冲区非空的条件变量 */

pthread_cond_t notfull; /* 缓冲区未满的条件变量 */

};

/* 初始化缓冲区结构 */

void init(struct prodcons *b)

{

pthread_mutex_init(&b->lock, NULL);

pthread_cond_init(&b->notempty, NULL);

pthread_cond_init(&b->notfull, NULL);

b->readpos = 0;

b->writepos = 0;

}

/* 将产品放入缓冲区,这里是存入一个整数*/

void put(struct prodcons *b, int data)

{

pthread_mutex_lock(&b->lock);

/* 等待缓冲区未满*/

if ((b->writepos + 1) % BUFFER_SIZE == b->readpos)

{

pthread_cond_wait(&b->notfull, &b->lock);

}

/* 写数据,并移动指针 */

b->buffer[b->writepos] = data;

b->writepos++;

if (b->writepos > = BUFFER_SIZE)

b->writepos = 0;

/* 设置缓冲区非空的条件变量*/

pthread_cond_signal(&b->notempty);

pthread_mutex_unlock(&b->lock);

}

/* 从缓冲区中取出整数*/

int get(struct prodcons *b)

{

int data;

pthread_mutex_lock(&b->lock);

/* 等待缓冲区非空*/

if (b->writepos == b->readpos)

{

pthread_cond_wait(&b->notempty, &b->lock);

}

/* 读数据,移动读指针*/

data = b->buffer[b->readpos];

b->readpos++;

if (b->readpos > = BUFFER_SIZE)

b->readpos = 0;

/* 设置缓冲区未满的条件变量*/

pthread_cond_signal(&b->notfull);

pthread_mutex_unlock(&b->lock);

return data;

}



/* 测试:生产者线程将1 到10000 的整数送入缓冲区,消费者线

程从缓冲区中获取整数,两者都打印信息*/

#define OVER ( - 1)

struct prodcons buffer;

void *producer(void *data)

{

int n;

for (n = 0; n \n", n);

put(&buffer, n);

} put(&buffer, OVER);

return NULL;

}



void *consumer(void *data)

{

int d;

while (1)

{

d = get(&buffer);

if (d == OVER)

break;

printf("--->%d \n", d);

}

return NULL;

}



int main(void)

{

pthread_t th_a, th_b;

void *retval;

init(&buffer);

/* 创建生产者和消费者线程*/

pthread_create(&th_a, NULL, producer, 0);

pthread_create(&th_b, NULL, consumer, 0);

/* 等待两个线程结束*/

pthread_join(th_a, &retval);

pthread_join(th_b, &retval);

return 0;

}
 




  5.WIN32、VxWorks、Linux线程类比



  目前为止,笔者已经创作了《基于嵌入式操作系统VxWorks的多任务并发程序设计》(《软件报》2006年5~12期连载)、《深入浅出Win32多线程程序设计》(天极网技术专题)系列,我们来找出这两个系列文章与本文的共通点。



   看待技术问题要瞄准其本质,不管是Linux、VxWorks还是WIN32,其涉及到多线程的部分都是那些内容,无非就是线程控制和线程通信,它们的许多函数只是名称不同,其实质含义是等价的,下面我们来列个三大操作系统共同点



详细表单:

事项 WIN32 VxWorks Linux
线程创建 CreateThread taskSpawn pthread_create
线程终止 执行完成后退出;线程自身调用ExitThread函数即终止自己;被其他线程调用函数TerminateThread函数 执行完成后退出;由线程本身调用exit退出;被其他线程调用函数taskDelete终止 执行完成后退出;由线程本身调用pthread_exit 退出;被其他线程调用函数pthread_cance终止
获取线程ID GetCurrentThreadId taskIdSelf pthread_self
创建互斥 CreateMutex semMCreate pthread_mutex_init
获取互斥 WaitForSingleObject、WaitForMultipleObjects semTake pthread_mutex_lock
释放互斥 ReleaseMutex semGive phtread_mutex_unlock
创建信号量 CreateSemaphore semBCreate、semCCreate sem_init
等待信号量 WaitForSingleObject semTake sem_wait
释放信号量 ReleaseSemaphore semGive sem_post



   6.小结



  本章讲述了Linux下多线程的控制及线程间通信编程方法,给出了一个生产者/消费者的实例,并将Linux的多线程与WIN32、VxWorks多线程进行了类比,总结了一般规律。鉴于多线程编程已成为开发并发应用程序的主流方法,学好本章的意义也便不言自明。

 

分享到:
评论

相关推荐

    Linux下C开发之线程通信

    Linux下C开发之线程通信 在Linux平台下,线程通信是C开发中一个非常重要的方面。本文将详细介绍Linux下C开发之线程通信的相关知识点,包括线程概念、线程创建、线程控制、线程通信和线程同步等。 一、线程概念 在...

    Linux下C语言开发集锦

    本集锦聚焦于Linux环境下的C语言开发,旨在帮助开发者深入理解和掌握C语言在Linux平台上的应用。 首先,Linux下C语言开发的基础知识包括对GCC(GNU Compiler Collection)的使用。GCC是Linux中的标准编译器,可以将...

    linux下C语言开发笔记整理

    网络编程是Linux下C语言开发的核心内容之一,涉及套接字编程,用于实现网络通信。了解TCP/IP协议栈以及如何使用套接字API(如`socket`、`bind`、`listen`、`accept`、`connect`、`send`和`recv`)进行网络数据的发送...

    Linux开发之线程通信

    【Linux开发之线程通信】深入解析 在Linux操作系统中,多线程编程是一个重要的概念,尤其对于系统级和应用级软件开发来说至关重要。虽然Linux内核最初并未直接实现线程模型,而是通过轻量级进程(LWP,Lightweight ...

    Linux下C语言应用编程

    Linux下C语言应用编程》深入浅出地讲解了Linux下C应用程序开发所需的基本工具、知识和技巧,主要包括:开发工具的使用和技巧、文件I/O编程、多进程编程、进程间通信、多线程编程、网络编程等内容。可以让读者轻松、...

    linux下c语言多线程网页爬虫源代码

    1. **Linux编程环境**:Linux提供了丰富的命令行工具和开源库,如glibc,用于支持C语言开发。GCC(GNU Compiler Collection)是Linux下的主要编译器,可以用来编译C语言源代码。 2. **C语言编程**:C语言是一种底层...

    linux下c语言开发

    在Linux环境下进行C语言开发是一项基础且重要的技能,无论你是初学者还是经验丰富的开发者,都能从中受益匪浅。Linux提供了丰富的开发工具和强大的系统调用,使得C语言编程变得更加高效和灵活。以下是一些关于...

    完全手册Linux系统下C程序开发详解 pdf

    《Linux系统下C程序开发详解》是一本专为在Linux环境下进行C语言编程设计的全面指南,由徐磊等专家撰写,旨在帮助读者深入理解并掌握在Linux操作系统中使用C语言进行程序开发的各项技能。这本书以清晰易懂的方式阐述...

    (linux下C语言实现)简易web服务器

    在Linux环境下,使用C语言编写一个简易的Web服务器是一项具有挑战性的任务,它涉及到网络编程、HTTP协议理解和操作系统层面的知识。下面将详细讲解这个项目所涵盖的关键知识点。 首先,我们来了解一下C语言在网络...

    Linux下C语言应用编程(作者-杨铸)配套教学ppt

    - `11-Linux下C语言编程环境.ppt`可能涵盖了如何在Linux上设置开发环境,包括安装GCC编译器,使用Makefile进行项目管理,以及调试工具GDB的使用。 2. **文件IO编程**: - `12-文件IO编程.ppt`会讲解Linux下的文件...

    linux下c语言学习

    在Linux下进行C语言开发,需要一些辅助工具,例如: - **编辑器**:如Vim、Emacs。 - **版本控制系统**:如Git。 - **构建工具**:如Autotools、CMake。 - **包管理工具**:如apt、yum。 这些工具可以帮助开发者更...

    Linux下C语言应用编程代码范例

    在Linux操作系统中,C语言是一种常用的编程语言,用于开发系统级程序、库和各种应用程序。这份"Linux下C语言应用编程代码范例"是学习和...通过深入理解和实践这些例子,你能够更熟练地运用C语言开发在Linux上的应用。

    Linux下的多进程/线程网络通信

    本作业要求在Linux下使用C语言实现一个抓取并分析网络数据的软件系统。该系统采用客户端和服务器端开发模式。在客户端,设计两个进程P1和P2同时读取网卡上的数据,P1读取数据后,负责将TCP报文中的头部数据写入到...

    基于Linux C语言的多线程模拟智能家具服务器源代码

    在本项目中,我们关注的是一个基于Linux操作系统,使用C语言编写的多线程智能家具服务器的源代码。这个服务器程序旨在模拟智能家居环境中的设备控制,可能是为了学习、实验或开发目的。以下是对这个项目中涉及的技术...

    Linux下C语言编程教程

    #### 第九章 Linux下的C开发工具介绍 - **编辑器**: 比较不同的文本编辑器,如vim、emacs等。 - **调试器**: 详细介绍GDB的使用方法。 - **版本控制**: 介绍Git等版本控制系统的基本用法。 - **构建工具**: 除了...

    贪吃蛇 linux开发 C语言 linux系统编程

    在IT行业中,Linux系统编程是开发者们不可或缺的技能之一,特别是在服务器端开发和嵌入式领域。本项目“贪吃蛇 Linux开发 C语言 Linux系统编程”旨在为已经掌握一定C语言基础的学习者提供一个有趣的实践平台,通过...

    Linux下C语言应用编程-作者-杨铸-配套ppt-源代码-教学大纲

    通过这些资源,学生不仅能学习到C语言的基础,还能掌握在Linux环境下进行系统级编程的技能,包括进程管理、线程编程和网络通信,这些都是现代软件开发中不可或缺的部分。这些知识对于想要从事Linux系统开发或者...

    linux下C语言函数手册

    本手册详细介绍了在Linux环境下使用C语言编程时所涉及的各种函数,这对于理解和开发高效、可靠的系统级程序至关重要。这份"Linux下C语言函数手册"Word版,将为学习者提供一个全面的参考指南,帮助他们掌握C语言在...

Global site tag (gtag.js) - Google Analytics