linux系统编程:线程原语

线程原语

线程概念

线程(thread),有时被称为轻量级进程(Lightweight Process,LWP),是程序执行流的最小单元。一个标准的线程由线程ID,当前指令指针(PC),寄存器集合和堆栈组成。更多详细解释看百度百科:线程

在Linux shell下通过命令 $ ps -Lf pid 查看指定pid号下的所有线程。

线程之间的共享与非共享

这里的线程是指同一进程下的线程。

共享:

1.文件描述符表
   2.每种信号的处理方式
   3.当前工作目录
   4.用户ID和组ID
   5.内存地址空间

非共享:

1.线程id
   2.处理器现场和栈指针(内核栈)
   3.独立的栈空间(用户空间栈)
   4.errno变量
   5.信号屏蔽字
   6.调度优先级

线程原语

通过命令 $ man -k pthread 查看系统下与线程有关的所有函数。

一般的,把main函数称作主线程或主控线程,其它线程都称作是子线程。

创建线程

#include<pthread.h>
int pthread_create(pthread_t *thread, const pthread_attr_t *attr, void *(*start_routine)(void *), void *arg);

pthread_t *thread:传递一个pthread_t变量地址进来,用于保存新线程的tid(线程ID)

const pthread_attr_t *attr:线程属性设置,如使用默认属性,则传NULL

void *(*start_routine)(void *):函数指针,指向新线程应该加载执行的函数模块

void *arg:指定线程将要加载调用的那个函数的参数

返回值:成功返回0,失败返回错误号。以前学过的系统函数都是成功返回0,失败返回-1,而错误号保存在全局变量errno中,而pthread库的函数都是通过返回值返回错误号,虽然每个线程也都有一个errno,但这是为了兼容其它函数接口而提供的,pthread库本身并不使用它,通过返回值返回错误码更加清晰。
Compile and link with -lpthread.
typedef unsigned long int pthread_t;   //在不同的系统中pthread_t可能会有不同的实现

结束线程

如果需要只终止某个线程而不终止整个进程,可以有三种方法:

1.return。这种方法对主控线程不适用,从main函数return相当于调用exit。

2.一个线程可以调用pthread_cancel终止同一进程中的另一个线程。

#include <pthread.h>
int pthread_cancel(pthread_t thread);

被取消的线程,退出值,定义在Linux的pthread库中常数PTHREAD_CANCELED的值是-1。可以在头文件pthread.h中找到它的定义:#define PTHREAD_CANCELED ((void *) -1)。也就是说,被pthread_cancel结束的线程,退出值,都是-1。

同一进程的线程间,pthread_cancel向另一线程发终止信号。系统并不会马上关闭被取消线程,只有在被取消线程下次系统调用时,才会真正结束线程。或调用pthread_testcancel,让内核去检测是否需要取消当前线程。

3.线程可以调用pthread_exit终止自己。

#include <pthread.h>
void pthread_exit(void *retval);
void *retval:线程退出时传递出的参数,可以是退出值或地址,如是地址时,不能是线程内部申请的局部地址。

在任何地方调用exit,都会使整个进程退出。

线程id

#include <pthread.h>
pthread_t pthread_self(void);
RETURN VALUE
This function always succeeds, returning the calling thread’s ID.

在线程中使用pthread_self,获取线程id。

回收线程

#include <pthread.h>
int pthread_join(pthread_t thread, void **retval);
pthread_t thread:回收线程的tid
void **retval:接收退出线程传递出的返回值
返回值:成功返回0,失败返回错误号

同进程类似,线程退出后,同样需要进行回收,所以pthread_join类似于wait。调用该函数的线程将挂起等待,直到id为thread的线程终止。thread线程以不同的方法终止,通过pthread_join得到的终止状态是不同的,总结如下:

  1. 如果thread线程通过return返回,retval所指向的单元里存放的是thread线程函数的返回值。
  2. 如果thread线程被别的线程调用pthread_cancel异常终止掉,retval所指向的单元里存放的是常数PTHREAD_CANCELED。
  3. 如果thread线程是自己调用pthread_exit终止的,retval所指向的单元存放的是传给pthread_exit的参数。

如果对thread线程的终止状态不感兴趣,可以传NULL给retval参数。

代码示例

使用return和pthread_exit结束进程

//thread_exit.c
#include <stdio.h>
#include <unistd.h>
#include <pthread.h>
void *fun_1(void *argv)
{
	printf("Hello %s,my thread id is %x\n", (char*)argv, pthread_self());
	sleep(1);
	//pthread_exit((void*)1);
	return (void*)1;
}
void *fun_2(void *argv)
{
	printf("Hello %c, my thread id is %x\n", (char)argv, pthread_self());
	sleep(2);
	pthread_exit((void*)"David");
}
int main(void)
{
	int *ret1;
	char *ret2;
	pthread_t tid1, tid2;
	pthread_create(&tid1, NULL, fun_1, (void*)"zhangxiang");
	pthread_create(&tid2, NULL, fun_2, (void*)‘D‘);
	pthread_join(tid1, (void*)&ret1);
	pthread_join(tid2, (void*)&ret2);
	printf("thread %x exit with %x\n", tid1, ret1);
	printf("thread %x exit with %s\n", tid2, ret2);
	return 0;
}
$ gcc thread_exit.c -lpthread
$ ./a.out
Hello D, my thread id is 745e9700
Hello zhangxiang, my thread id is 74fea700
thread 74fea700 exit with 1
thread 745e9700 exit with David

使用pthread_cancel终止另一个线程

//thread_cancel.c
#include <stdio.h>
#include <pthread.h>
void *fun(void *argv)
{
	printf("thread %x running\n", pthread_self());
	pthread_exit((void*)0);
}
int main(void)
{
	pthread_t tid;
	void *ret;
	pthread_create(&tid, NULL, fun, NULL);
	pthread_join(tid, &ret);
	printf("thread %x exit with %d\n", tid, (int)ret);
	pthread_create(&tid, NULL, fun, NULL);
	pthread_cancel(tid);
	pthread_join(tid, &ret);
	printf("thread %x exit with %d\n", tid, (int)ret);
	return 0;
}
$ gcc thread_cancel.c -lpthread
$ a.out
thread 96da9700 running
thread 96da9700 exit with 0
thread 96da9700 running
thread 96da9700 exit with -1

既可以在主线程中使用pthread_cancel终止子线程,也可以在子线程中终止另一个子线程。

CCPP Blog 目录

版权声明:本文为博主原创文章,转载,请注明出处。

时间: 2024-10-18 10:35:23

linux系统编程:线程原语的相关文章

Linux系统编程——线程同步与互斥:无名信号量

信号量概述 信号量广泛用于进程或线程间的同步和互斥,信号量本质上是一个非负的整数计数器,它被用来控制对公共资源的访问. 编程时可根据操作信号量值的结果判断是否对公共资源具有访问的权限,当信号量值大于 0 时,则可以访问,否则将阻塞.PV 原语是对信号量的操作,一次 P 操作使信号量减1,一次 V 操作使信号量加1. 信号量主要用于进程或线程间的同步和互斥这两种典型情况. 信号量用于互斥: 信号量用于同步: 在 POSIX 标准中,信号量分两种,一种是无名信号量,一种是有名信号量.无名信号量一般用

Linux系统编程——线程私有数据

在多线程程序中,经常要用全局变量来实现多个函数间的数据共享.由于数据空间是共享的,因此全局变量也为所有线程共有. 测试代码如下: #include <stdio.h> #include <pthread.h> #include <unistd.h> #include <stdlib.h> int key = 100; //全局变量 void *helloworld_one(void *arg) { printf("the message is %s

linux系统编程--线程同步

同步概念 所谓同步,即同时起步,协调一致.不同的对象,对“同步”的理解方式略有不同. 如,设备同步,是指在两个设备之间规定一个共同的时间参考: 数据库同步,是指让两个或多个数据库内容保持一致,或者按需要部分保持一致: 文件同步,是指让两个或多个文件夹里的文件保持一致.等等 而,编程中.通信中所说的同步与生活中大家印象中的同步概念略有差异.“同”字应是指协同.协助.互相配合.主旨在协同步调,按预定的先后次序运行. 线程同步 同步即协同步调,按预定的先后次序运行. 线程同步,指一个线程发出某一功能调

Linux系统编程——线程池

线程池基本原理 在传统服务器结构中,常是有一个总的监听线程监听有没有新的用户连接服务器,每当有一个新的用户进入,服务器就开启一个新的线程用户处理这 个用户的数据包.这个线程只服务于这个用户,当用户与服务器端关闭连接以后,服务器端销毁这个线程.(关于并发服务器更多详情,请看<并发服务器>). 然而频繁地开辟与销毁线程极大地占用了系统的资源,而且在大量用户的情况下,系统为了开辟和销毁线程将浪费大量的时间和资源.线程池提供了一个解决外部大量用户与服务器有限资源的矛盾. 线程池和传统的一个用户对应一个

Linux系统编程——线程同步与互斥:互斥锁

为什么需要互斥锁? 在多任务操作系统中,同时运行的多个任务可能都需要使用同一种资源.这个过程有点类似于,公司部门里,我在使用着打印机打印东西的同时(还没有打印完),别人刚好也在此刻使用打印机打印东西,如果不做任何处理的话,打印出来的东西肯定是错乱的. 下面我们用程序模拟一下这个过程,线程一需要打印" hello ",线程二需要打印" world ",不加任何处理的话,打印出来的内容会错乱: #include <stdio.h> #include <

Linux系统编程——线程同步与互斥:读写锁

当有一个线程已经持有互斥锁时,互斥锁将所有试图进入临界区的线程都阻塞住.但是考虑一种情形,当前持有互斥锁的线程只是要读访问共享资源,而同时有其它几个线程也想读取这个共享资源,但是由于互斥锁的排它性,所有其它线程都无法获取锁,也就无法读访问共享资源了,但是实际上多个线程同时读访问共享资源并不会导致问题. 在对数据的读写操作中,更多的是读操作,写操作较少,例如对数据库数据的读写应用.为了满足当前能够允许多个读出,但只允许一个写入的需求,线程提供了读写锁来实现.... www.worlduc.com/

嵌入式 Linux系统编程(六)——系统信息

嵌入式 Linux系统编程(六)--系统信息 一.时间 Linux系统下常用的时间类型:time_t.struct tm.struct timeval.struct timespec. 1.time_t类型时间 time_t实际是一个长整型.其值表示为从UTC(coordinated universal time)时间1970年1月1日00时00分00秒(也称为Linux系统的Epoch时间)到当前时刻的秒数.由于time_t类型长度的限制,它所表示的时间不能晚于2038年1月19日03时14分

Linux系统编程(第2版)笔记 (本书基本上就是Linux C API的简单使用说明,入门级别的)

Linux系统编程(第2版) 跳转至: 导航. 搜索 目录 1 入门和基本概念 2 文件I/O 3 缓冲I/O 4 高级文件I/O 5 进程管理 6 高级进程管理 7 线程 8 文件和目录管理 9 内存管理 10 信号 11 时间(这里谈不上系统编程了,就是C库API) 12 附录A C语言的GCC扩展 13 附录B 参考书目 入门和基本概念 文件I/O read(): EINTR EAGAIN 其他错误:EBADF EFAULT EINVAL EIO Append模式:每次write之前的文件

Linux系统编程之进程

前一段时间对文件I/O的基本操作基本操作做了总结,今天这里继续按照我的理解对linux系统编程的进程操作进行总结. 首先我们先理解几个概念:程序.进程.线程. 所谓程序,就是计算机指令的集合,它以文件的形式存储在磁盘上,进程是一个程序在其自身的地址空间中的一次执行活动.而线程进程内的一个执行单元,也是进程内的可调度实体.说完这个不知道大家理解了吗?反正我第一次听到这个概念以后看到的时候可以明白过后就忘记了,现在我给大家举一个例子帮助大家理解,大家都看电视剧吧,所谓程序,就是一个剧本,像什么<西游

Linux系统编程【转】

转自:https://blog.csdn.net/majiakun1/article/details/8558308 一.Linux系统编程概论 1.1 系统编程基石 syscall: libc:标准C库.系统调用封装.线程库.基本应用工具 gcc: 1.2 模块接口 API:应用程序编程接口,源代码级别,能通过编译,由标准C语言定义,libc来实现 ABI:应用程序二进制接口,二进制级别,能正常运行,关注调用约定.字节序.寄存器使用.系统调用.链接.二进制格式等,很难实现 1.3 错误处理 <