




已阅读5页,还剩11页未读, 继续免费阅读
版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
Linux下多线程编程详解线程(thread)技术早在60年代就被提出,但真正应用多线程到操作系统中去,是在80年代中期,solaris是这方面的佼佼者。传统的Unix也支持线程的概念,不过在一个进程(process)中只允许有一个线程,这样多线程就意味着多进程。目前,多线程技术已被许多操作系统所支持,包括视窗系统/NT,当然,也包括Linux。为什么有了进程的概念后,还要再引入线程呢?使用多线程到底有哪些好处?什么的系统应该选用多线程?我们首先必须回答这些问题。使用多线程的理由之一是和进程相比,他是一种非常节俭的多任务操作方式。我们知道,在Linux系统下,启动一个新的进程必须分配给他独立的地址空间,建立众多的数据表来维护他的代码段、堆栈段和数据段,这是一种昂贵的多任务工作方式。而运行于一个进程中的多个线程,他们彼此之间使用相同的地址空间,共享大部分数据,启动一个线程所花费的空间远远小于启动一个进程所花费的空间,而且,线程间彼此转换所需的时间也远远小于进程间转换所需要的时间。据统计,总的说来,一个进程的开销大约是个线程开销的30倍左右,当然,在具体的系统上,这个数据可能会有较大的差别。使用多线程的理由之二是线程间方便的通信机制。对不同进程来说,他们具有独立的数据空间,要进行数据的传递只能通过通信的方式进行,这种方式不仅费时,而且非常不方便。线程则不然,由于同一进程下的线程之间共享数据空间,所以一个线程的数据能直接为其他线程所用,这不仅快捷,而且方便。当然,数据的共享也带来其他一些问题,有的变量不能同时被两个线程所修改,有的子程式中声明为static的数据更有可能给多线程程式带来灾难性的打击,这些正是编写多线程程式时最需要注意的地方。除了以上所说的好处外,不和进程比较,多线程程式作为一种多任务、并发的工作方式,当然有以下的好处:1) 提高应用程式响应。这对图像界面的程式尤其有意义,当一个操作耗时非常长时,整个系统都会等待这个操作,此时程式不会响应键盘、鼠标、菜单的操作,而使用多线程技术,将耗时长的操作(time consuming)置于一个新的线程,能避免这种尴尬的情况。2) 使多CPU系统更加有效。操作系统会确保当线程数不大于CPU数目时,不同的线程运行于不同的CPU上。3) 改善程式结构。一个既长又复杂的进程能考虑分为多个线程,成为几个独立或半独立的运行部分,这样的程式会利于理解和修改。下面我们先来尝试编写一个简单的多线程程式。简单的多线程编程Linux系统下的多线程遵循POSIX线程接口,称为pthread。编写Linux下的多线程程式,需要使用头文件pthread.h,连接时需要使用库libpthread.a。顺便说一下,Linux下pthread的实现是通过系统调用clone()来实现的。clone()是Linux所特有的系统调用,他的使用方式类似fork,关于clone()的周详情况,有兴趣的读者能去查看有关文件说明。下面我们展示一个最简单的多线程程式 pthread_create.c。一个重要的线程创建函数原型:#include int pthread_create(pthread_t *restrict tidp,const pthread_attr_t *restrict attr, void *(*start_rtn)(void),void *restrict arg); 返回值:若是成功建立线程返回0,否则返回错误的编号 形式参数: pthread_t *restrict tidp 要创建的线程的线程id指针 const pthread_attr_t *restrict attr 创建线程时的线程属性 void* (start_rtn)(void) 返回值是void类型的指针函数 void *restrict arg start_rtn的行参 例程1: 功能:创建一个简单的线程 程式名称:pthread_create.c /* Name:pthread_create.c* Used to study the multithread programming in Linux OS* Author:zeickey* Date:2006/9/16 * Copyright (c) 2006,All Rights Reserved!*/#include #include void *myThread1(void) int i; for (i=0; i#include #include void *create(void *arg) int *num; num=(int *)arg; printf(create parameter is %d n,*num); return (void *)0;int main(int argc ,char *argv) pthread_t tidp; int error; int test=4; int *attr=&test; error=pthread_create(&tidp,NULL,create,(void *)attr); if(error) printf(pthread_create is created is not created . n); return -1; sleep(1); printf(pthread_create is created .n); return 0; 编译方法:gcc -lpthread pthread_int.c -Wall 执行结果:create parameter is 4pthread_create is created is created . 例程总结: 能看出来,我们在main函数中传递的整行指针,传递到我们新建的线程函数中。 在上面的例子能看出来我们向新的线程传入了另一个线程的int数据,线程之间还能传递字符串或是更复杂的数据结构。例程3: 程式功能:向新建的线程传递字符串 程式名称:pthread_string.c/* Name:pthread_string.c* Used to study the multithread programming in Linux OS* Pass a char* parameter to the thread.* Author:zeickey* Date:2006/9/16 * Copyright (c) 2006,All Rights Reserved!*/#include #include #include void *create(void *arg) char *name; name=(char *)arg; printf(The parameter passed from main function is %s n,name); return (void *)0;int main(int argc, char *argv) char *a=zieckey; int error; pthread_t tidp; error=pthread_create(&tidp, NULL, create, (void *)a); if(error!=0) printf(pthread is not created.n); return -1; sleep(1); printf(pthread is created. n); return 0; 编译方法:gcc -Wall pthread_string.c -lpthread 执行结果:The parameter passed from main function is zieckey pthread is created. 例程总结: 能看出来main函数中的字符串传入了新建的线程中。例程4: 程式功能:向新建的线程传递字符串 程式名称:pthread_struct.c/* Name:pthread_struct.c* Used to study the multithread programming in Linux OS* Pass a char* parameter to the thread.* Author:zeickey* Date:2006/9/16 * Copyright (c) 2006,All Rights Reserved!*/#include #include #include #include struct menber int a; char *s;void *create(void *arg) struct menber *temp; temp=(struct menber *)arg; printf(menber-a = %d n,temp-a); printf(menber-s = %s n,temp-s); return (void *)0;int main(int argc,char *argv) pthread_t tidp; int error; struct menber *b; b=(struct menber *)malloc( sizeof(struct menber) ); b-a = 4; b-s = zieckey; error = pthread_create(&tidp, NULL, create, (void *)b); if( error ) printf(phread is not created.n); return -1; sleep(1); printf(pthread is created.n); return 0; 编译方法:gcc -Wall pthread_struct.c -lpthread 执行结果:menber-a = 4 menber-s = zieckey pthread is created. 例程总结: 能看出来main函数中的一个结构体传入了新建的线程中。 线程包含了标识进程内执行环境必须的信息。他集成进程中的所有信息都是对线程进行共享的,包括文本程式、程式的全局内存和堆内存、栈及文件描述符。 例程5: 程式目的:验证新建立的线程能共享进程中的数据 程式名称:pthread_share.c /* Name:pthread_share_data.c* Used to study the multithread programming in Linux OS* Pass a char* parameter to the thread.* Author:zeickey* Date:2006/9/16 * Copyright (c) 2006,All Rights Reserved!*/#include #include #include static int a=4;void *create(void *arg) printf(new pthread . n); printf(a=%d n,a); return (void *)0;int main(int argc,char *argv) pthread_t tidp; int error; a=5; error=pthread_create(&tidp, NULL, create, NULL); if(error!=0) printf(new thread is not create . n); return -1; sleep(1); printf(new thread is created . n); return 0; 编译方法:gcc -Wall pthread_share_data.c -lpthread 执行结果:new pthread . a=5 new thread is created . 例程总结:能看出来,我们在主线程(main函数)更改了我们的全局变量a的值的时候,我们新建立的线程则打印出来了改动的值,能看出能访问线程所在进程中的数据信息。 2、线程的终止 如果进程中所有一个线程中调用exit,_Exit,或是_exit,那么整个进程就会终止, 和此类似,如果信号的默认的动作是终止进程,那么,把该信号发送到线程会终止进程。 线程的正常退出的方式: (1) 线程只是从启动例程中返回,返回值是线程中的退出码 (2) 线程能被另一个进程进行终止 (3) 线程自己调用pthread_exit函数 两个重要的函数原型:#include void pthread_exit(void *rval_ptr);/*rval_ptr 线程退出返回的指针*/int pthread_join(pthread_t thread,void *rval_ptr); /*成功结束进程为0,否则为错误编码*/ 例程6 程式目的:线程正常退出,接受线程退出的返回码 程式名称:pthread_exit.c/* Name:pthread_exit.c* Used to study the multithread programming in Linux OS* A example showing a thread to exit and with a return code.* Author:zeickey* Date:2006/9/16 * Copyright (c) 2006,All Rights Reserved!*/#include #include #include void *create(void *arg) printf(new thread is created . n); return (void *)8;int main(int argc,char *argv) pthread_t tid; int error; void *temp; error = pthread_create(&tid, NULL, create, NULL); if( error ) printf(thread is not created . n); return -1; error = pthread_join(tid, &temp); if( error ) printf(thread is not exit . n); return -2; printf(thread is exit code %d n, (int )temp); return 0; 编译方法:gcc -Wall pthread_exit.c -lpthread 执行结果:new thread is created .thread is exit code 8 例程总结:能看出来,线程退出能返回线程的int数值。线程退出不仅仅能返回线程的int数值,还能返回一个复杂的数据结构。 例程7 程式目的:线程结束返回一个复杂的数据结构 程式名称:pthread_return_struct.c#include #include #include struct menber int a; char *b;temp=8,zieckey;void *create(void *arg) printf(new thread . n); return (void *)&temp;int main(int argc,char *argv) int error; pthread_t tid; struct menber *c; error = pthread_create(&tid, NULL, create, NULL); if( error ) printf(new thread is not created . n); return -1; printf(main . n); error = pthread_join(tid,(void *)&c); if( error ) printf(new thread is not exit . n); return -2; printf(c-a = %d n,c-a); printf(c-b = %s n,c-b); sleep(1); return 0; 编译方法:gcc -Wall pthread_return_struct.c -lpthread 执行结果:main .new thread .c-a = 8c-b = zieckey例程总结:一定要记得返回的数据结构要是在这个数据要返回的结构没有释放的时候应用,如果数据结构已发生变化,那返回的就不会是我们所需要的,而是脏数据3、线程标识 函数原型: #include pthread_t pthread_self(void);pid_t getpid(void); getpid()用来取得目前进程的进程识别码,函数说明 例程8 程式目的:实目前新建立的线程中打印该线程的id和进程id 程式名称:pthread_id.c /* Name:pthread_id.c* Used to study the multithread programming
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 梁弯曲时的强度计算
- 2025年麻醉科临床麻醉应急处理模拟测试卷答案及解析
- 民族复兴是我的责任课件
- 2025年肝脏疾病乙肝病毒感染的防治与护理模拟考试卷答案及解析
- 民族团结课件教育
- 民族团结花课件
- 2025年普外科急性胃粘膜损伤的处理模拟考试卷答案及解析
- 2025年精神科护理技巧与团体治疗模拟测试卷答案及解析
- 2025年精神科常见病例诊疗技术考核答案及解析
- 2025年放射电影学临床诊断综合考试答案及解析
- 尿道损伤专题知识讲座
- 长沙会战优质获奖课件
- GB/T 254-2022半精炼石蜡
- 校本课程篆刻教学设计
- GB/T 20967-2007无损检测目视检测总则
- GB/T 12220-2015工业阀门标志
- 当代世界经济与政治第二章课件
- PS考试试题及答案
- 新都区文化产业发展建议报告
- 时代邻里4度°服务美学品质关怀体系
- 养老机构行政值班查房记录表格
评论
0/150
提交评论