线程
创建一个缺省的线程
缺省的线程的属性:
l 非绑定
l 未分离
l 一个缺省大小的堆栈
l 具有和父线程一样的优先级
用 phread_attr_init() 创建一个缺省的属性对象,
用属性对象创建一个线程 pthread_create(3T)
int p thread_create ( pthread_t *tid, const pthread_attr_t *tattr, void *(*start_routine)(void*), void *arg );
#include<pthread.h>
pthread_attr_t tattr;
pthread_t tid;
extern void *start_routine(void *arg);
void *arg;
int ret;
/*default behavior*/
ret = pthread_create( &tid, NULL, start_routine, arg );
/*init with default attributes*/
ret = pthread_attr_init( &tattr );
/*default behavior specified*/
ret = pthread_create( &tid, &tattr, start_routine, arg );
tattr 中含有初始化线程所需的属性,值赋为 NULL 即缺省。 start_routine 是线程入口函数的起始地址。当 start_routine 返回时,相应的线程就结束了。线程结束时的退出状态值是 start_routine 函数用 phread_exit() 函数返回的返回值。当 pthread_create() 函数调用成功时,线程标识符保存在参数 tid 指针指向中。
返回值, pthread_create() 成功后返回 0 ,
EAGAIN 超过了某个限制,如 LWPs 过多。
EINVAL tattr 值非法。
创建子线程时,传给子线程的输入参数最好是 malloc() 返回的指针(这样的指针指向进程堆中的存储空间)或指向全局变量的指针,而不要是指向局部变量的指针。因为当子线程访问输入参数时,创建子线程的函数可能已结束,局部变量也就不存在了。
等待线程结束
pthread_join(3T)
int pthread_join( pthread_t tid, void **status );
#include<pthread.h>
pthread_t tid;
int ret;
int status;
/*waiting to join thread “tid” with status*/
ret = pthread_join( tid, &status );
/*waiting to join thread “tid” without status*/
ret = pthread_join( tid, NULL );
pthread_join() 会阻塞调用它的线程,直到参数 tid 指定的线程结束。 tid 指定的线程必须在当前进程中,且必须是非分离的。 status 接收指定线程终止时的返回状态码。
不能有多个线程等待同一个线程终止,否则返回错误码 ESRCH
当 pthread_join() 返回时,终止线程占用的堆栈等资源已被回收。
返回值:成功返回 0
ESRCH tid 指定的线程不是一个当前线程中合法且未分离的线程。
EDEADLK tid 指定的是当前线程。
EINVAL tid 非法。
分离一个线程
pthread_detach(3T)
将非分离线程设置为分离线程。
int pthread_detach( pthread_t tid );
#include<pthread.h>
pthread_t tid;
int ret;
ret = pthread_detach( tid );
该函数通知线程库,当线程终止以后, tid 所指线程的内存可以被立即收回。
返回值:成功返回 0
EINVAL tid 所指线程不是一个合法线程。
ESRCH tid 指定的线程不是一个当前线程中合法且未分离的线程。
为线程数据创建一个键
多线程的 c 语言程序具有三种数据:局部变量,全局变量,线程数据( TSD )
TSD 类似于全局变量,但是线程私有的 。
每个 TSD 都有个键同他相关联。
pthread_key_create(3T)
int pthread_key_create( pthread_key_t *key, void (*destructor)(*void) );
#include<pthread.h>
pthread_key_t key;
int ret;
/*key create without destructor*/
ret = pthread_key_create( &key, NULL );
/*key create with destructor*/
ret = pthread_key_destructor( &key, destructor );
该函数成功时,份配的建放在 key 中,必须保证 key 指向的内存区有效 。 destructor 用来释放不再需要的内存。
返回值:成功返回 0
EAGAIN key 名字空间耗尽
ENOMEM 没有足够的内存空间创建一个新的键。
删除线程数据的键
pthread_key_delete(3T)
solaris 线程接口中没有该函数
int pthread_key_delete ( pthread_key_t key );
#include<pthread.h>
pthread_key_t key;
int ret;
ret = pthread_key_delete( key );
在调用该函数之前,必须释放和本线程相关联的资源, pthread_key_delete() 不会引发键的解析函数。
返回值:成功返回 0
EINVAL key 值非法
设置线程数据键
pthread_setspecific(3T)
设置和某个线程数据键绑定在一起的线程数据 (一般是指针)
函数将指向专有数据的指针value 设置进 由key指示的结构体 中;
int pthread_setspecific ( pthread_key_t key, const void *value );
#include<pthread.h>
pthread_key_t key;
void *value;
int ret;
ret = pthread_setspecific( key, value );
返回值:成功返回 0
ENOMEM 没有足够的虚拟内存
EINVAL key 值非法
pthread_setspecific() 不释放原来绑定在键上的内存,给一个键绑定新的指针时,必须释放原指针指向的内存 ,否则会发生内存泄漏。
获取线程数据键
pthread_getspecific(3T)
获取绑定在线程数据键上的值,并在指定的位置存储值
返回存放在对应结构体中的专有指针;
int pthread_getspecific( pthread_key_t key, void**value )
#include<pthread.h>
pthread_key_t key;
void *value;
pthread_getspecific( key, &value );
返回值:不返回错误码
取线程标识符
pthread_self(3T)
取当前线程的标识符
pthread_t pthread_self( void );
#include<pthread.h>
pthread_t tid;
tid = pthread_self();
返回值:当前线程标识符。
比较线程标识符
pthread_equal(3T)
int pthread_equal( pthread_t tid1, pthread_t tid2 );
#include<pthread.h>
pthread_t tid1,tid2
int ret;
ret = pthread_equal( tid1, tid2 );
返回值:如果 tid1 和 tid2 相同返回非零值,否则返回 0 。如果参数非法,返回值不可预知。
初始化线程
pthread_once(3T)
用来调用初始化函数,只有第一次调用有效。
int pthread_once( pthread_once_t *once_control, void(*init_routine)(void) );
#include<pthread.h>
pthread_once_t once_control = PTHREAD_ONCE_INIT;
int ret;
ret = pthread_once( &once_control, init_routine );
once_control 界定了相应的初始化函数是否被调用过。
返回值:成功返回 0
EINVAL 某个参数为 NULL
出让当前线程对处理器的控制权
sched_yeild(3R)
把当前线程的优先权让给有相同或更高优先级的线程。
int sched_yeild( void );
#include<pthread.h>
int ret;
ret = sched_yeild();
返回值:成功返回 0
ENOSYS 当前版本不支持 sched_yield()
设置线程的优先级
pthread_setschedparam(3T)
int pthread_setschedparam( pthread_t tid, int policy, const struct sched_param *param );
#include<pthread.h>
pthread_t tid;
int ret;
struct sched_param param;
int priority;
/*sched_priority will be the priority of the thread*/
sched_param,sched_priority = priority;
/*only supported policy ,other will result in ENOTSUP*/
policy = SCHED_OTHER;
/*scheduling parameters of target thread*/
ret = pthread_setschedparam( tid, policy, ¶m );
返回值:成功返回 0
EINVAL 属性值非法
ENOTSUP 属性值在当前版本不支持
取线程的优先级
pthread_getschedparam(3T)
int pthread_getschedparam( pthread_t tid, int policy, struct schedparam *param );
#include<pthread.h>
pthread_t tid;
sched_param param;
int prioprity;
int policy;
int ret;
/*scheduling parameters of target thread*/
ret = pthread_getschedparam( tid, &policy, ¶m );
/*sched_priority contains the priority of the thread*/
priority = param.sched_priority;
返回值:成功返回 0
ESRCH tid 不是一个现存的线程。
向线程发信号
pthread_kill(3T)
int pthread_kill( pthread_t tid, int sig );
#include<pthread.h>
#include<signal.h>
int sig;
pthread_t tid;
int ret;
ret = pthread_kill( tid, sig );
tid 指定的线程必须和函数当前线程在同一个进程中。
sig 为 0 时,进行错误检查,不发送信号,往往被用来检查 tid 的合法性。
返回值:成功返回 0
EINVAL sig 不是合法信号量
ESRCH tid 不是当前进程中的线程
访问当前线程的信号掩码
pthread_sigmask(3T)
int pthread_sigmask( int how, const sigset_t *new, sigset_t *old );
#include<pthread.h>
#include<signal.h>
int ret;
sigset_t old, new;
ret = pthread_sigmask( SIG_SETMASK, &new, &old );
ret = pthread_sigmask( SIG_BLOCK, &new, &old );
ret = pthread_sigmask( SIG_UNBLOCK, &new, &old );
how 表示对当前信号掩码进行什么操作。
SIG_SETMASK :在信号掩码中加入 new 信号集, new 表示新增加的要屏蔽的信号。
SIG_BLOCK :在信号掩码中删去 new 信号集, new 表示新增加的不需再屏蔽的信号。
SIG_UNBLOCK :用 new 信号集替换信号掩码, new 表示所有需要屏蔽的信号。
当 new 为 NULL 时,无论 how 是什么,当前线程的信号掩码都不会改变。
旧的信号掩码保存在 old 中。
返回值:成功返回 0
EINVAL how 的值未定义
安全的复制
pthread_atfork(3T)
int pthread_atfork( void(*prepare)(void), void(*parent)(void), void(*child)(void) );
终止线程
pthread_exit(3T)
void pthread_exit(void *status);
#include<pthread.h>
int status;
pthread_exit( &status );
终止当前线程,所有绑定在线程键上的内存将释放。如果当前线程是未分离的,该线程的标识符和推出代码( status )被保留,直到其它线程用 pthread_join() 等待当前线程的终止。如果当前线程是分离的, status 被忽略,线程标识符立即收回。
返回值:若 status 不为 NULL ,线程的退出代码被置为 status 指向的值。
一个线程可以用一下方式终止自身运行。
从线程的入口函数返回。
调用 pthread_exit()
用 POSIX 的 pthread_cancel()
退出可以用以下方式:
异步的
由线程库定义的一系列退出点
有应用程序定义的一系列退出点
退出点
由程序通过 pthread_testcancel() 建立
调用了 pthread_cond_wait() 或 pthread_cond_timedwait() 等待一个条件的线程
调用了 pthread_join() 等待另一个线程结束的线程。
被阻塞在 sigwait(2) 上的线程。
一些标准的库函数。
退出线程
pthread_cancel(3T)
int pthread_cancel( pthread_t thread );
#include<pthread.h>
pthread_t thread;
int ret;
ret = pthread_cancel ( thread ) ;
返回值:成功返回 0
ESRCH 无指定的线程。
允许或禁止退出
pthread_setcancelstate(3T)
缺省是允许退出的。
int pthread_setcancelstate( int state, int *oldstate );
#include<pthread.h>
int oldstate;
int ret;
/*enable*/
ret = pthread_setcancelstate( PTHREAD_CANCEL_ENABLE, &oldstate );
/*disabled*/
ret = pthread_setcancelstate( PTHREAD_CANCEL_DISABLE, &oldstate );
返回值:成功返回 0
EINVAL state 值非法
设置退出类型
pthread_setcanceltype(3T)
可以设置成延迟类型或异步类型。缺省是延迟类型。异步类型下,线程可以在执行中的任何时候被退出。
int pthread_setcanceltype( int type, int *oldtype );
#include<pthread.h>
int oldtype;
int ret;
/*deferred mode*/
ret = pthread_setcanceltype( PTHREAD_CANCEL_DEFERRED, &oldtype );
/*async mode*/
ret = pthread_setcanceltype( PTHREAD_CANCEL_ASYNCHRONOUS, &oldtype );
返回值:成功返回 0
EINVAL state 值非法
创建退出点
pthread_testcancel(3T)
void pthread_testcancel( void );
#include<pthread.h>
pthread_testcancel();
只有当线程的退出状态是允许退出,退出类型是延迟类型时,有效。
没有返回值。
将一个善后处理函数推入退出堆栈
pthread_cleanup_push(3T)
pthread_cleanup_pop(3T)
void Pthread_cleanup_push( void(*routine)(void*), void *args );
void pthread_cleanup_pop( int execute );
#include<pthread.h>
/*push the handler “routine” on cleanup stack*/
pthread_cleanup_push( routine, arg );
/*pop the “func” out of cleanup stack and execute “func”*/
pthread_cleanup_pop( 1 );
/*pop the “func” and don’t execute “func”*/
pthread_cleanup_pop( 0 );
1. 线程属性
只能在线程创建时制定属性,不能在运行时改变它。
一旦属性对象被配置好,它在进程范围内都是有效的。
初始化属性
pthread_attr_init(3T)
初始化一个线程属性对象,属性值是缺省值,占用内存由线程库分配。
int pthread_attr_init( pthread_attr_t *tattr );
#include<pthread.h>
pthread_attr_t tattr;
int ret;
/*initialize an attribute to the default value*/
ret = pthread_attr_init( &tattr );
属性对象的缺省值:
scope (线程域) PTHREAD_SCOPE_PROCESS
Detachstate (分离状态) PTHREAD_CREATE_JOINABLE
Stackaddr (堆栈地址) NULL
Stacksize (堆栈大小) 1Mb
priority (优先级) 父进程优先级
Inheritsched (继承调度优先级) PTHREAD_INHERIT_SCHED
schedpolicy (调度策略) SCHED_OTHER
返回值:成功返回 0
ENOMEM 没有足够的内存初始化线程属性对象
释放属性对象
pthread_attr_destroy(3T)
int pthread_attr_destroy( pthread_attr_t *tattr );
#include<pthread.h>
pthread_attr_t tattr;
int ret;
ret = pthread_attr_destroy( &tattr );
返回值:成功返回 0
EINVAL tattr 值非法
设置分离状态
pthread_attr.setdetachstate(3T)
创建线程时,如果指定这个线程为分离线程,一旦这个线程终止,他的线程标识符和其他相关的资源可以立即被使用。如果不需要等待某个线程终止,可以把它设定为分离。
int pthread_attr_setdetachstate( pthread_attr_t *tattr, int detachstate );
#include<pthread.h>
pthread_attr_t tattr;
int ret;
/*set the thread detach state*/
ret = pthread_attr_setdetachstate( &tattr, PTHREAD_CREATE_DETACHED );
返回值:成功返回 0
EINVAL tattr 的值或 detachstate 的值非法
取分离状态
pthread_attr_getdetachstate(3T)
int pthread_attr_getdetachstate( const pthread_attr_t *tattr, int *detachstate );
#include<pthread.h>
pthread_attr_t tattr;
int detachstate;
int ret;
ret = pthread_attr_getdetachstate( &tattr, &detachstate );
返回值:成功返回 0
EINVAL tattr 的值或 detachstate 的值非法
相关推荐
LINUX 线程函数大全 LINUX 线程函数大全是一个详细的线程函数手册,涵盖了线程创建、线程属性、线程结束、线程等待、线程分离、线程数据等多方面的知识点。本手册将详细讲解每个线程函数的使用方法、参数设置、...
这篇"LINUX线程函数大全[定义].pdf"文档详细介绍了在Linux环境下进行线程操作的关键函数,包括线程创建、线程等待、线程分离以及线程特定数据(Thread Specific Data, TSD)的管理。 首先,线程的创建是通过`...
线程函数是控制和管理这些线程的关键工具,这里我们将深入探讨Linux线程函数,包括创建、等待、结束和分离线程,以及线程特定数据(TSD)的管理。 首先,我们来看线程的创建。`pthread_create()`函数是创建新线程的...
本篇文章将深入探讨Linux C函数大全中的关键知识点,帮助读者掌握这些基础但至关重要的编程工具。 一、标准库函数 Linux C函数主要基于ANSI C标准库,即C99或C11标准。这些函数包括输入/输出处理、数学运算、内存...
"Linux函数大全"是一部非常实用的参考资料,它涵盖了广泛的功能,旨在帮助开发者在日常编程中快速查找和理解所需的函数。下面,我们将深入探讨一些重要的Linux函数及其用途。 1. **标准输入输出**: - `printf` 和...
这个“常用Linux函数库大全”涵盖了广泛的功能,对于学习和使用Linux系统的人来说,具有极高的参考价值。 首先,让我们了解一下Linux中的主要函数库: 1. **Glibc(GNU C Library)**:它是Linux系统中最基础的库...
《Linux函数大全》是针对Linux操作系统中编程时常用的函数进行详尽整理的参考资料。这个.chm文件,全称为“Compiled HTML Help”,是一种Windows系统下的帮助文档格式,它将多个HTML页面整合在一个文件中,便于用户...
本资源"Linux C函数参考大全"包含了丰富的C语言函数信息,特别关注于在Linux环境下进行字符串处理、文件操作、进程管理、线程操作以及网络通信等关键领域的函数用法。下面将对这些关键知识点进行详细解释。 1. **...
《Linux系统函数调用大全》是一本深入探讨Linux操作系统中函数调用的宝贵资源,它涵盖了广泛的主题,旨在帮助开发者和系统管理员更好地理解和利用Linux内核功能。这本书以PDF格式提供,是学习和查阅Linux系统编程的...
此外,还有`pthread`库中的线程操作函数,如`pthread_create()`和`pthread_join()`。 文件权限控制篇讲解了如何使用`chmod()`、`chown()`和`chgrp()`来改变文件的权限和所有者,以及`access()`用于检查文件访问权限...
这份"Linux C语言函数大全(htm 版,比较全)"很可能是一个HTML文档,提供了一个方便的在线查阅格式,适合程序员随时查询和学习。 C语言函数大全通常会涵盖以下几个主要部分: 1. **基本数据类型和控制结构**:...
多线程函数的实例教程 很经典很经典很经典很经典
5. **进程与线程**:Linux提供了一套丰富的系统调用来管理进程和线程,如fork()创建新进程,exec()系列函数执行新的程序,pthread_create()创建线程,pthread_join()等待线程结束等。 6. **信号处理**:通过...
本文将基于"Linux函数详解(函数大全)"这一主题,深入解析Linux C函数的使用和重要性,涵盖从基本输入输出到文件操作,再到进程控制等多个方面。 首先,我们从最基础的输入输出函数开始,如`printf`和`scanf`,...
"Linux C语言函数大全"这个资源很可能是针对C语言编程在Linux环境下的一个详细指南,涵盖了广泛的功能和实用技巧。下面,我们将深入探讨C语言在Linux中的重要性,以及一些常见的C语言函数和其在Linux系统编程中的...
Linux线程的挂起与恢复是一个复杂且需要谨慎处理的问题。通过互斥锁和条件变量,我们可以实现一种用户空间的模拟方式,但这并不总是一种安全或高效的方法。在设计多线程程序时,应优先考虑使用更安全的同步机制,如...
1. `pthread_create()`:用于创建新的线程,传入线程属性、线程函数和参数。 2. `pthread_join()`:等待指定线程结束,并回收其资源。 3. `pthread_exit()`:线程结束时调用,可以返回一个退出状态。 4. `pthread_...
"Linux C函数大全"是一个极佳的学习资源,它涵盖了Linux编程中C语言的广泛函数,帮助初学者快速理解和掌握这些函数的用法。 1. **标准库函数**: - `printf` 和 `scanf`:这是输入/输出处理的基本函数,用于格式化...