线程与进程关键字对比

创建新流 fork/pthread_create

退出控制流 exit/pthread_exit

获取退出状态 waitpid/pthread_join

在退出时的清理工作 atexit/pthread_cleanup_push

非正常退出 abort/pthread_cancel

创建线程

int pthread_create(线程ID返回值, 线程属性, 任务地址, 任务附加参数)

获取线程ID: pthread_t pthread_self(void)

判断是否同一线程: int pthread_equal(pthread_t tid1, pthread_t tid2)

线程终止

线程退出方式: pthread_exit; 线程正常执行完毕返回; 被同一进程中的其它线程取消

错误退出方式: 调用exit, _exit, _Exit都会使整个进程退出

获取返回值: pthread_join

线程清理

pthread_cleanup_push: 添加一个清理函数, 如果意外退出可以确保资源释放

pthread_cleanup_pop(0): 取消清理函数, 比如主动清理完成时便可以调用pop来取消push的操作

以下三种情况会触发push操作:

  • 调用pthread_exit退出(调用return不会触发)
  • 响应pthread_cancel请求
  • pthread_cleanup_pop(int), int参数为非零时

线程同步

1.互斥量

2.读写锁

3.条件变量

例子

1.创建和退出线程

#include <stdlib.h>
#include <stdio.h>
#include <pthread.h>
#include <unistd.h> void *
thr_fn1(void *arg){
printf("thread 1 returning \n");
return((void *)1);
} void *
thr_fn2(void *arg){
printf("thread 2 exiting\n");
pthread_exit((void *)2);
} int main(){
int err;
pthread_t tid1,tid2;
void *tret; err=pthread_create(&tid1,NULL,thr_fn1,NULL);
if(err != 0){
perror("pthread_create error");
return -1;
}
err=pthread_create(&tid2,NULL,thr_fn2,NULL);
if(err != 0){
perror("pthread_create error");
return -1;
} err=pthread_join(tid1,&tret);
if(err != 0){
perror("pthread_join error");
return -1;
}
printf("thread 1 exit code %d\n",(int)tret);
err=pthread_join(tid2,&tret);
if(err != 0){
perror("pthread_join error");
return -1;
}
printf("thread 2 exit code %d\n",(int)tret);
return 0;
}

2.清理函数

#include <stdlib.h>
#include <stdio.h>
#include <pthread.h>
#include <unistd.h> void cleanup(void *arg){
printf("in cleanup : %s\n",arg);
} void *
thr_fn1(void *arg){
puts("thread start");
pthread_cleanup_push(cleanup,"the first time");
pthread_cleanup_push(cleanup,"the second time");
if(1==(int)arg)
return((void*)1);
else if(2==(int)arg)
pthread_exit((void*)2);
pthread_cleanup_pop(0);
pthread_cleanup_pop(0);
return ((void *)0);
} int main(){
int err;
pthread_t tid1;
void *tret; //pthread_create第四个参数,
//0时会执行pop取消清理函数, 1时以return方式退出不触发清理函数,2时以pthread_exit方式退出触发清理函数
err=pthread_create(&tid1,NULL,thr_fn1,(void *)1);
if(err != 0){
perror("pthread_create error");
return -1;
} err=pthread_join(tid1,&tret);
if(err != 0){
perror("pthread_join error");
return -1;
}
printf("thread 1 exit code %d\n",(int)tret);
return 0;
}

3.互斥量

#include <stdlib.h>
#include <stdio.h>
#include <pthread.h>
#include <unistd.h>
#include <string.h> struct people{
char name[10];
int age;
int count;
pthread_mutex_t lock;
}; struct people * init(){
struct people *man;
if((man=malloc(sizeof(struct people))) != NULL){
man->count=1;
if(pthread_mutex_init(&man->lock,NULL) != 0){
free(man);
return NULL;
}
strcpy(man->name,"wahaha");
man->age=20;
}
return man;
} void add_count(struct people *man){
pthread_mutex_lock(&man->lock);
puts("----in add_count----");
printf("now count=%d\n",++man->count);
puts("----out add_count----");
pthread_mutex_unlock(&man->lock);
} void del_count(struct people *man){
pthread_mutex_lock(&man->lock);
if(--man->count == 0){
pthread_mutex_unlock(&man->lock);
pthread_mutex_destroy(&man->lock);
puts("last count to del");
free(man);
}else{
pthread_mutex_unlock(&man->lock);
}
} void * thr_fn(void *arg){
add_count(arg);
sleep(1);
del_count(arg);
} int main(){
struct people *man;
man=init();
if(man == NULL){
perror("man init error");
return -1;
} int err;
pthread_t tid1,tid2; err=pthread_create(&tid1,NULL,thr_fn,man);
if(err != 0){
perror("pthread_create error");
return -1;
}
err=pthread_create(&tid2,NULL,thr_fn,man);
if(err != 0){
perror("pthread_create error");
return -1;
} err=pthread_join(tid1,NULL);
if(err != 0){
perror("pthread_join error");
return -1;
}
err=pthread_join(tid2,NULL);
if(err != 0){
perror("pthread_join error");
return -1;
} del_count(man);
return 0;
}

linux 线程笔记的更多相关文章

  1. linux线程笔记1之创建线程

    1 线程与进程的对比 这里有一个笔记详细的阐述 http://blog.csdn.net/laviolette/article/details/51506953 2 创建线程函数 int pthrea ...

  2. Dubbo入门到精通学习笔记(十一):Dubbo服务启动依赖检查、Dubbo负载均衡策略、Dubbo线程模型(结合Linux线程数限制配置的实战分享)

    文章目录 Dubbo服务启动依赖检查 Dubbo负载均衡策略 Dubbo线程模型(结合Linux线程数限制配置的实战分享) 实战经验分享( ** 属用性能调优**): Dubbo服务启动依赖检查 Du ...

  3. Linux线程互斥学习笔记--详细分析

    一.互斥锁 为啥要有互斥? 多个进程/线程执行的先后顺序不确定,何时切出CPU也不确定. 多个进程/线程访问变量的动作往往不是原子的. 1. 操作步骤 (1)创建锁 // 创建互斥锁mutex pth ...

  4. linux线程的实现

    首先从OS设计原理上阐明三种线程:内核线程.轻量级进程.用户线程 内核线程 内核线程就是内核的分身,一个分身可以处理一件特定事情.这在处理异步事件如异步IO时特别有用.内核线程的使用是廉价的,唯一使用 ...

  5. 跟着鸟哥学Linux系列笔记3-第11章BASH学习

    跟着鸟哥学Linux系列笔记0-扫盲之概念 跟着鸟哥学Linux系列笔记0-如何解决问题 跟着鸟哥学Linux系列笔记1 跟着鸟哥学Linux系列笔记2-第10章VIM学习 认识与学习bash 1. ...

  6. linux线程的实现【转】

    转自:http://www.cnblogs.com/zhaoyl/p/3620204.html 首先从OS设计原理上阐明三种线程:内核线程.轻量级进程.用户线程 内核线程 内核线程就是内核的分身,一个 ...

  7. Linux 读书笔记 二

    一.实验说明 1. 环境登录 无需密码自动登录,系统用户名shiyanlou,密码shiyanlou 若不小心登出后,直接刷新页面即可 2. 环境使用 完成实验后可以点击桌面上方的“实验截图”保存并分 ...

  8. 【转载】linux内核笔记之进程地址空间

    原文:linux内核笔记之进程地址空间 进程的地址空间由允许进程使用的全部线性地址组成,在32位系统中为0~3GB,每个进程看到的线性地址集合是不同的. 内核通过线性区的资源(数据结构)来表示线性地址 ...

  9. # linux读书笔记(3章)

    linux读书笔记(3章) 标签(空格分隔): 20135328陈都 第三章 进程管理 3.1 进程 进程就是处于执行期的程序(目标码存放在某种存储介质上).但进程并不仅仅局限于一段可执行程序代码( ...

随机推荐

  1. ASP.NET页面与IIS底层交互和工作原理详解(第三回)

    引言 Http 请求处理流程 和 Http Handler 介绍 这两篇文章里,我们首先了解了Http请求在服务器端的处理流程,随后我们知道Http请求最终会由实现了IHttpHandler接口的类进 ...

  2. JQuery POST请求乱码...

    引言: 在JQuery的Ajax POST请求中,进行请求,其中的中文在后台,显示为乱码,该如何解决呢? 问题的引入: var regid = $('#oregion').combobox('getV ...

  3. iframe跨域自适应高度

    思路: 现有主界面main在域a下,被嵌套页面B在域b下,被嵌套页面B又嵌套一个在域a下的中介页面A. 当用户打开浏览器访问mail.html的时候载入B,触发B的onload事件获取其自身高度,然后 ...

  4. JavaScript 键盘event.keyCode值列表大全

      JavaScript 键盘event.keyCode值列表大全   event.keyCode值列表大全,对于需要根据键盘按键触发相应事件的朋友需要. 网上收集的KeyCode值方便大家查找: k ...

  5. js、jquery、动态添加option项

    var selid = document.getElementById("sltid"); for(var i=0; i<10;i++){     //循环添加多个值 sid ...

  6. linux初学 :linux 常用命令(二)

    压缩和解压命令 gzip/guzip   zip/unzip   tar gzip和gunzip一般可用参数是-r,例: gzip test.txt 压缩文件 gzip -r test 压缩所有tes ...

  7. java执行程序

    执行jar java -jar x.jar 执行jar里边指定class java -cp x.jar x.x.x

  8. EasyGUI的安装

    1.下载EasyGUI 在官方网站上下载http://easygui.sourceforge.net/将安装包下载下来,放到桌面上并解压. 2.安装EasyGUI ①在开始菜单的搜索中输入cmd,打开 ...

  9. php读取mysql中文数据出现乱码

    1.PHP页面语言本身的编码类型不合适,这时候,你直接在脚本中写的中文肯定是乱码,不用说数据库了:    解决方法:选择'UTF8'或者'gb2312',这样客户浏览器会自动选择并出现正确的中文显示. ...

  10. MapView

    有两种方式可以将 MapView 添加到应用当中:一个是 XML 方式,另一个是硬编码方式:一般多采用 XML方式,方便调整布局及其属性相关设置. 1.1XML方式 <com.esri.andr ...