POSIX多线程之创建线程pthread_create && 线程清理pthread_cleanup
pthreads定义了一套C程序语言类型、函数、与常量。以pthread.h和一个线程库实现。
数据类型:
pthread_t:线程句柄
pthread_attr_t:线程属性
线程操作函数:
pthread_create():创建一个线程
pthread_exit():终止当前线程
pthread_cancel():中断另外一个线程的运行
pthread_join():阻塞当前的线程,直到另外一个线程运行结束
pthread_attr_init():初始化线程的属性
pthread_attr_setdetachstate():设置脱离状态的属性
pthread_attr_getdetachstate():获取脱离状态的属性
pthread_attr_destroy():删除线程的属性
pthread_kill():向线程发送一个信号
.....
创建线程:
int pthread_create(pthread_t *thread, const pthread_attr_t *attr, void *(*start_rtn) (void*),void *arg);
*thread:线程id (指向线程标识符的指针)
*attr:线程属性(通常为空)
*start_rtn:线程要执行的函数(线程运行函数的起始地址)
*arg:start_rtn的参数
返回值为零才算成功,其它表示错误
编译时需要加上 -lpthread库
示例一:
#include <stdio.h>
#include <pthread.h>
void *pthread_1(void) //线程1执行的函数
{
inti;
for(i=1; i < 5; i++){
printf("pthread1.\n");
sleep(1);
}
return (void *)0;
}
void *pthread_2(void) //线程2执行的函数
{
inti;
for(i=1;i < 5; i++){
printf("pthread2.\n");
sleep(1);
}
return (void *)0;
}
int main(int argc, char argv[])
{
intret = 0;
pthread_tpthid1, pthid2; //线程的id
ret= pthread_create(&pthid1,NULL, (void *)pthread_1, NULL);
if(ret) // 非0则创建失败
{
perror("createthread 1 failed.\n");
return1;
}
ret= pthread_create(&pthid2, NULL, (void *)pthread_2, NULL);
if(ret)
{
perror("createthread 2 failed.\n");
return1;
}
pthread_join(pthid1,NULL); //等待pthid1,否则不会等待,继续往下执行
pthread_join(pthid2,NULL); //如无,遇到下面的return会结束进程
return0;
}
示例二:传递参数,初始化
/*
*pthread_init.c
*
* Created on: Jan 20, 2013
* Author: linuxdba@qq.com
*/
#include <stdio.h>
#include <pthread.h>
void *create(void *arg)
{
int *num;
num = (int*)arg;
printf("theinput number is: %d.\n", *num);
return(void *)0;
}
int main(int argc, char argv[])
{
intret;
intt;
printf("Inputone number:");
scanf("%d",&t);
int*attr = &t;
pthread_tpthid;
ret= pthread_create(&pthid, NULL, (void *)create, (void *)attr);
if(ret)
{
perror("createthread failed.\n");
return1;
}
pthread_join(pthid,NULL);
return0;
}
示例三:共享数据
/*
*pthread_share.c
*
* Created on: Jan 20, 2013
* Author: linuxdba@qq.com
*/
#include <stdio.h>
#include <pthread.h>
static int i = 3;
void *pthread_1(void)
{
printf("thread1, i is: %d.\n", i);
return(void *)0;
}
void *pthread_2(void)
{
printf("thread2, i is: %d.\n", i);
return(void *)0;
}
int main(int argc, char argv[])
{
intret = 0;
inti = 5;
pthread_tpthid1, pthid2;
printf("mainthread, i is: %d.\n", i);
ret= pthread_create(&pthid1, NULL, (void *)pthread_1, NULL);
if(ret)
{
printf("createthread 1 failed.\n");
return1;
}
ret= pthread_create(&pthid2, NULL, (void *)pthread_2, NULL);
if(ret)
{
printf("createthread 2 failed.\n");
return1;
}
pthread_join(pthid1,NULL);
pthread_join(pthid2,NULL);
return0;
}
此段程序之中两个线程是对等的,不存在先后顺序;
i 为全局变量,main函数内部为局部的变量。
http://www.kernel.org/doc/man-pages/online/pages/man3/pthread_create.3.html
http://baike.baidu.com/view/974776.htm
线程清理:pthread_cleanup_push() & pthread_cleanup_pop()
A cancellation clean-up handler is poppedfrom the stack and executed in
The following circumstances:(三种情况会执行线程清理)
1. When a thread is canceled, allof the stacked clean-up handlers are popped and executed in the reverse of theorder in which they were pushed onto the stack. (线程取消)
2. When a thread terminates bycallingpthread_exit(3),all clean-up handlers are executed as described in the preceding point. (Clean-up handlers are not called if thethread terminates by performing a return from the thread start function.) (push与pop之间执行了pthread_exit)
3. When a thread calls pthread_cleanup_pop()with a nonzero execute argument, the top-most clean-up handler is popped andexecuted.(pop内为非零参数)
示例代码:
/*
*pthread_push_pop.c
*
* Created on: Jan 21, 2013
* Author: linuxdba@qq.com
*/
#include <stdio.h>
#include <pthread.h>
void *clean1(void *arg)
{
printf("clean1func: %s\n", arg);
return(void *)0;
}
void *clean2(void *arg)
{
printf("clean2func: %s\n", arg);
return(void *)0;
}
void *thread(void )
{
pthread_tthid;
thid= pthread_self(); //获取线程ID
printf("thethread is: %d.\n",thid);
printf("theprocess id is: %d.\n", getpid()); //进程ID
pthread_cleanup_push((void*)clean1,"first push");
pthread_cleanup_push((void*)clean2, "second push");
printf("Inthread.\n");
pthread_exit("threadexit.\n"); //退出线程,会执行push指定函数
/*return NULL; */ //不会执行push指定函数
pthread_cleanup_pop(0);//为0,只有在异常退出时才会执行push指定的函数,pop第一个
pthread_cleanup_pop(1);//为非0,即便是正常,也会执行push指定的函数,pop第二个
returnNULL;
}
int main(int argc, char argv[])
{
intret = 0;
pthread_tpthid;
ret= pthread_create(&pthid, NULL,(void *)thread,NULL);
if(ret)
{
printf("createthread failed.\n");
return-1;
}
printf("theprocess is: %d.\n", getpid()); //获取进程ID
pthread_join(pthid,NULL);
}
运行结果:
the process is: 3893.
the thread is:1738999552.
the process id is: 3893.
In thread.
clean2 func: second push
clean1 func: first push
push与pop采用栈方式,先进后出,后进先出,所以clean2会在前面。
需要注意的几点:
1) 线程属于进程的,只有一个进程ID,一样的;
2) push和pop是采用栈方式;
3) pthread_cleanup_pop后面为0则执行push指定函数,非0则即便正常也执行
4) push与pop之间代码若异常退出,包括pthread_exit,会进行相应的函数处理,但是若其间执行了return,则直接结束线程,不执行push指定函数。
参考文档:
http://man7.org/linux/man-pages/man3/pthread_cleanup_push.3.html
POSIX多线程之创建线程pthread_create && 线程清理pthread_cleanup的更多相关文章
- C++ Linux 多线程之创建、管理线程
线程就是,在同一程序同一时间内同意运行不同函数的离散处理队列. 这使得一个长时间去进行某种特殊运算的函数在运行时不阻碍其它的函数变得十分重要. 线程实际上同意同一时候运行两种函数,而这两个函数不必相互 ...
- 7、Java并发性和多线程-如何创建并运行线程
以下内容转自http://ifeve.com/creating-and-starting-java-threads/: Java线程类也是一个object类,它的实例都继承自java.lang.Thr ...
- Java多线程的创建(一)
方法一:继承Thread类实现 1.创建一个类A,并继承Thread类 2.重写A的run()方法 3.创建A的实例对象b,即创建了线程对象 4.使用b调用start()方法:启动线程(会自动调用ru ...
- POSIX多线程—线程基本概念
http://blog.csdn.net/livelylittlefish/article/details/7957007 作者:阿波链接:http://blog.csdn.net/livelylit ...
- iOS开发多线程篇—创建线程
iOS开发多线程篇—创建线程 一.创建和启动线程简单说明 一个NSThread对象就代表一条线程 创建.启动线程 (1) NSThread *thread = [[NSThread alloc] in ...
- POSIX多线程——基本线程管理函数介绍
POSIX基本的几个线程管理函数见下表: ------------------------------------------------------------------------------- ...
- posix多线程--线程取消
1.三种取消状态Off 禁用取消Deferred 推迟取消:在下一个取消点执行取消Asynchronous 异步取消:可以随时执行取消 in ...
- posix多线程--三种基本线程编程模型
本文介绍了三种构建线程解决方案的方式. 一.流水线:每个线程执行同一种操作,并把操作结果传递给下一步骤的线程. 代码示例如下:终端输入一个int值,每个线程将该值加1,并将结果传给下一个线程. #in ...
- java多线程-概念&创建启动&中断&守护线程&优先级&线程状态(多线程编程之一)
今天开始就来总结一下Java多线程的基础知识点,下面是本篇的主要内容(大部分知识点参考java核心技术卷1): 1.什么是线程以及多线程与进程的区别 2.多线程的创建与启动 3.中断线程和守护线程以及 ...
随机推荐
- 【Python】解析Python中的异常操作
目录结构: contents structure [-] try,except,else,finally块 异常处理 使用except而不带任何异常类型 使用except而带多种异常类型 try-fi ...
- 为什么java里面经常作List判断的时候,既要判断list不为null,又要判断size>0呢?
没有考虑到具体的问题上面,我们单纯的来讲: 为什么java里面经常作List判断的时候,既要判断list不为null,又要判断size>0呢? list == null 说明list没有初始化( ...
- pycharm安装、首次使用及汉化
摘自:https://blog.csdn.net/qq_30463497/article/details/84992480 一.下载pycharm安装包打开PyCharm的官方下载地址:http:// ...
- nginx安装和命令
1. nginx安装 1.1 mac上安装 brew search nginx brew install nginx 1.2 windows上安装 下载nginx.zip,解压到D盘,发送快捷方式到桌 ...
- 在nginx环境下搭建https服务,代理到本地web项目
安装过程略. 1.证书准备 本地调试,可以安装自签名证书,安装方法参考https本地自签名证书添加到信任证书访问 2.修改配置文件 将上面的配置文件拷贝到conf目录,添加或者修改节点如下 http{ ...
- 使用Cloud Toolkit部署SpringBoot项目到服务器
由于我们经常发布项目到测试服,在测试服上调试一些本地无法调试的东西,所以出现了各种打包,然后上传.启动,时间都耗费在这无聊的事情上面了,偶然在网上看到IntelliJ IDEA有 Cloud Tool ...
- kafka高吞吐量的分布式发布订阅的消息队列系统
一:kafka介绍kafka(官网地址:http://kafka.apache.org)是一种高吞吐量的分布式发布订阅的消息队列系统,具有高性能和高吞吐率. 1.1 术语介绍BrokerKafka集群 ...
- 【剑指offer】数组在排序数组中出现的次数
题目描述 统计一个数字在排序数组中出现的次数. 分析:数组有序,采用二分查找无疑 两种方法,时间复杂度差不多,都是利用二分查找,不过统计k出现的次数有所不同而已 方法1:二分查找k,找到任意一个k的下 ...
- drf源码系列
过滤器 对查询出来的数据进行筛选可写可不写 from rest_framework.filters import BaseFilterBackend 源码 ''' def filter_queryse ...
- day02——while、字符串格式化、运算符、编码初识
day02 while--关键字(死循环) 格式:while 条件: 循环体 print(1) while True: print("痒") print("鸡你太美& ...