1.线程标识
#include <pthread.h>
int pthread_equal(pthread_t tid1, pthread_t tid2); //如果相等返回非0值,否则返回0.
线程可以通过调用pthread_self函数获得自身线程ID。
#include<pthread.h>
pthread_t pthread_self(void); //返回值为调用线程的线程ID。
2.线程的创建
#include<pthread.h>
int pthread_create(pthread_t *restrict tidp, const pthread_attr_t *restrict attr, void *(*start_rtn)(void*), void*restrict arg );
//若成功则返回0,否则返回错误编号
当pthread_create成功返回时,由tidp指向的内存单元被设置为新创建线程的线程ID,attr参数用于定制各种不同的线程
3.线程终止
#include<pthread.h>
void pthread_exit(void *rval_ptr);
rval_ptr是一个无类型指针,与传给启动例程的单个参数类型,进程中的其他线程可以通过调用pthread_join函数访问
#include <pthread.h>
int pthread_join(pthread_t thread, void **rval_ptr); //若成功则返回0,否则返回错误编号。
调用线程将一直阻塞,知道指定的线程调用pthread_exit、从启动例程中返回或者取消。如果线程只是从它的启动例程
#include <stdio.h>
#include <pthread.h>
#include <string.h> void* thr_fn1(void *arg){
printf("thread 1 return.\n");
return (void*)1;
} void* thr_fn2(void *arg){
printf("thread 2 exit.\n");
pthread_exit((void*)2);
} int main(void){
int err;
pthread_t tid1,tid2;
void* tret; err = pthread_create(&tid1, NULL, thr_fn1, NULL);
if(err != 0){
printf("pthread_create:%s\n",strerror(err));
return -1;
} err = pthread_create(&tid2, NULL, thr_fn2, NULL);
if(err != 0){
printf("pthread_create:%s\n",strerror(err));
return -1;
} err = pthread_join(tid1, &tret);
if(err != 0){
printf("pthread_join:%s\n",strerror(err));
return -1;
}
printf("thread1 exit code %d\n",(int)tret); err = pthread_join(tid2, &tret);
if(err != 0){
printf("pthread_join:%s\n",strerror(err));
return -1;
}
printf("thread1 exi2 code %d\n",(int)tret); return 0;
}
thread 1 return.
thread1 exit code 1
thread1 exi2 code 2
#include<pthread.h>
int pthread_cancel(pthread_t itd); //若成功则返回0,否则返回错误编号
#include<pthread.h>
void pthread_cleanup_push(void (*rtn)(void *), void* arg);
void pthread_cleanup_pop(int execute);
当线程执行以下动作时调用清理函数,调用参数为arg,清理函数rtn的调用顺序是由pthread_clean_push函数来安排的。
#include <stdio.h>
#include <pthread.h> void cleanup(void *arg){
printf("cleanup:%s\n",(char*)arg);
} void *thr_fn1(void *arg){
printf("thread 1 start\n");
pthread_cleanup_push(cleanup, "thread 1 first handler");
pthread_cleanup_push(cleanup, "thread 1 second handler");
printf("thread 1 push complete\n");
return (void *)1;
pthread_cleanup_pop(0);
pthread_cleanup_pop(0);
} void *thr_fn2(void *arg){
printf("thread 2 start\n");
pthread_cleanup_push(cleanup, "thread 2 first handler");
pthread_cleanup_push(cleanup, "thread 2 second handler");
printf("thread 2 push complete\n");
pthread_exit((void *)2);
pthread_cleanup_pop(0);
pthread_cleanup_pop(0);
} int main(void){
pthread_t tid1,tid2;
int err;
void* tret; err = pthread_create(&tid1, NULL, thr_fn1, (void*)1);
if(err != 0){
printf("pthread_create:%s\n",strerror(err));
return -1;
} err = pthread_create(&tid2, NULL, thr_fn2, (void*)2);
if(err != 0){
printf("pthread_create:%s\n",strerror(err));
return -1;
} err = pthread_join(tid1,&tret);
if(err != 0){
printf("pthread_join:%s\n",strerror(err));
return -1;
}
printf("thread 1 exit code:%d\n",(int)tret); err = pthread_join(tid2,&tret);
if(err != 0){
printf("pthread_join:%s\n",strerror(err));
return -1;
}
printf("thread 2 exit code:%d\n",(int)tret); return 0;
}
thread 2 push complete
cleanup:thread 2 second handler
cleanup:thread 2 first handler
thread 1 start
thread 1 push complete
thread 1 exit code:1
thread 2 exit code:2
#include <pthread.h>
int pthread_detach(pthread_t tid); //成功则返回1,否则返回错误编号。
实践:
#include <stdio.h>
#include <pthread.h>
#include <string.h> void *thr_fn(void *arg){
printf("thread start\n");
printf("thread complete\n");
pthread_exit((void *)1);
} int main(void){
pthread_t tid1;
int err;
void* tret; err = pthread_create(&tid1, NULL, thr_fn, (void*)1);
if(err != 0){
printf("pthread_create:%s\n",err);
return -1;
} err = pthread_detach(tid1);
if(err != 0){
printf("pthread_detach:%s\n",err);
return -1;
} err = pthread_join(tid1,&tret);
if(err != 0){
printf("pthread_join:%s\n",strerror(err));
return -1;
}
printf("thread 1 exit code:%d\n",(int)tret);
return 0;
}
《UNIX环境高级编程》笔记--线程的标识、创建和终止的更多相关文章
-
【UNIX环境高级编程】线程同步
当多个线程共享相同的内存时,需要确保每个线程看到一致的数据视图.如果每个线程使用的变量都是其他线程不会读取和修改的,那么就不存在一致性问题.同样,如果变量是只读的也不会有一致性问题.但是,当一个线程可 ...
-
UNIX环境高级编程笔记之文件I/O
一.总结 在写之前,先唠几句,<UNIX环境高级编程>,简称APUE,这本书简直是本神书,像我这种小白,基本上每看完一章都是“哇”这种很吃惊的表情.其实大概三年前,那会大三,我就买了这本书 ...
-
Unix环境高级编程(十一)线程
一个进程在同一时刻只能做一件事情,线程可以把程序设计成在同一时刻能够做多件事情,每个线程处理各自独立的任务.线程包括了表示进程内执行环境必需的信息,包括进程中标识线程的线程ID.一组寄存器值.栈.调度 ...
-
UNIX环境高级编程笔记之线程
本章涉及到线程的一些基本知识点,讨论了现有的创建线程和销毁线程的POSIX.1原语,此外,重点介绍了线程同步问题,讨论了三种基本的同步机制:互斥量.读写锁.条件变量.
-
UNIX环境高级编程笔记 目录
每一章的重点会使用加粗字体 第一章:UNIX基础知识:UNIX体系结构:文件和目录:输入和输出:程序和进程:出错处理:信号:时间值:系统调用和库函数 第三章:文件I/O:文件描述符:文件操作函数:文件 ...
-
UNIX环境高级编程笔记之高级I/O
本章说明了很多高级I/O功能: 非阻塞I/O——发一个I/O操作,不使其阻塞,记录锁,STREAMS机制 I/O多路转接——select和poll函数 readv和writev函数,以及存储映射I/O ...
-
UNIX环境高级编程笔记之进程控制
本章重点介绍了进程控制的几个函数:fork.exec族._exit.wait和waitpid等,主要需要掌握的是父进程和子进程之间的运行机制,怎么处理进程的正常和异常终止.以及怎么让进程执行不同的程序 ...
-
UNIX环境高级编程笔记之进程环境
本章讲的都是一些非常基础的知识,目的是为了下一章讲进程控制做铺垫,所以,本章就不做过多的总结了,直接看图吧.
-
UNIX环境高级编程笔记之标准I/O库
一.总结 文件I/O一章讲了不带缓冲的I/O,本章讲的是带缓冲的I/O.不带缓冲针对的是内核的系统调用,而带缓冲针对的是用户空间的标准库函数,是基于带缓冲的I/O实现的.不带缓冲的I/O通过文件描述符 ...
-
(九) 一起学 Unix 环境高级编程 (APUE) 之 线程
. . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...
随机推荐
-
《PHP Manual》阅读笔记2
本次笔记截止到 语言参考 流程控制. 1.没有结合的相同优先级的运算符不能连在一起使用,例如 1 < 2 > 1 在PHP是不合法的.但另外一方面表达式 1 <= 1 == 1 是合 ...
-
hdu 2075
PS:水得不能再水..刚开始还以为是大数..要用到快速幂...谁知道想太多...就普通int型.. 代码: #include "stdio.h" int main(){ int a ...
-
SSM框架
1.http://www.cnblogs.com/verlen11/p/5349747.html 2.Mybatis http://www.cnblogs.com/xdp-gacl/p/4261895 ...
-
关于http与https区别
http与https: http叫超文本传输协议,信息为明文传输.https是具有安全性的传输协议,是由http+ssl层,需要到ca申请证书,一般需要费用.信息为加密传输,需要验证用户身份.二者的端 ...
-
[LeetCode] Average of Levels in Binary Tree 二叉树的层平均值
Given a non-empty binary tree, return the average value of the nodes on each level in the form of an ...
-
sqlserver 删除表中 指定字符串
源表T "单据编号" "航班计划日期" "航班号" "起飞航站代码&q ...
-
ElasticSearch6.1.1集群搭建
其实早就想研究ES了,因为之前用solr,资料较少(这倒不是问题,有问题去官网读文档),貌似用的人比较少?(别打我)前几天去京东面试,我觉得有必要了解一下es,昨天晚上简单了解了官方文档,今天居然鼓捣 ...
-
css小笔记
一.优先级 important>内联>id>class = 属性 = 伪类 >标签 = 伪元素 > 通配符(*) important声明 1,0,0,0 ID选择器 0, ...
-
文件处理----Properties文件处理
properties是一种属性文件,这种文件以key=value格式存储内容,代码中可以使用Properties类来读取这个文件,然后得到数据. 当配置文件用,由于难以表达层次,复杂点可以使用xml做 ...
-
[UE4]封装、继承、多态
面向对象编程的三大特征 一.封装 公开能做什么,隐藏如何做.封装的目的是减少类之间的依赖. 二.继承 让一个类拥有另一个类的状态和行为,前者可以不加修改地完全复用后者的实现,也可以对有些行为做出自己的 ...