C语言由浅入深讲解线程的定义

目录
  • 线程的概念
  • 线程的创建
  • 线程的终止
  • 线程标识的比较
  • 线程的取消
  • 线程等待
  • 线程分离

线程的概念

可以简单理解为一个正在独立运行的函数

注:

1.posix线程是一套标准吗,而不是实现

2.线程标识: phread_t,可能是整形也可能是结构体指针等

*简单介绍关于线程标识的函数*
 *pthread_equarl() ;判断两个线程标识是否相等*
*pthread_self();返回自身的线程标识*

线程的创建

pthread_creat();

int pthread_create(

pthread_t *restrict tidp, //新创建的线程ID指向的内存单元。

const pthread_attr_t *restrict attr, //线程属性,默认为NULL

void *(*start_rtn)(void *), //新创建的线程从start_rtn函数的地址开始运行

void *restrict arg //默认为NULL。若上述函数需要参数,将参数放入结构中并将地址作为arg传入。);

线程的调度取决于调度器策略

# include<stdlib.h>
# include<pthread.h>
static void* func(void* p)
{
 puts("thread is working");
 return NULL;
}
int main()
{
  pthread_t tid;
  int err;
  puts("begin!");
  err = pthread_create(&tid,NULL,func,NULL);
  if(err)
  {
   exit(-1);
  }
  puts("end!");
  exit(0);
}

线程的终止

3种方式 :

1)线程从启动历程返回,返回值就是线程的退出码

2)线程可以被同一进程中的其他线程取消

3)线程可以调用 pthread)exit()函数

线程退出函数

void pthread_exit(void *value_ptr);

线程标识的比较

获取线程号

pthread_t pthread_self(void);

比较

int pthread_equal(pthread_t t1, pthread_t t2);

线程的取消

int pthread_cancel(pthread_t thread);

取消有两种状态 : 允许和不允许

允许取消又分为: 异步cancel , 推迟 cancel(默认) -> 推迟到cancel点

cancel点: posix定义的canceldian 都是可能引发阻塞的系统调用

pthread_setcancelstate(int );//设置是否允许取消

pthread_setcanceltype(); //设置取消方式 异步还是推迟

注:pthread_cancel 调用具有具有一定的延时性,因为cancel点: posix定义的canceldian 都是可能引发阻塞的系统调用,并不会立即被处理,不建议当线程立即被创建后立即进行cancel取消(线程创建,并不会立即被调度);也不建议在线程退出前执行线程cancel取消(线程可能在取消之前就已经退出了);建议在线程执行中进行cancel取消线程

void pthread_testcancel(void); //本函数为设置取消点

当不关心线程的返回信息时可以直接使用线程分离函数进行回收

反之,需要获取线程的返回值时,可以使用线程等待

线程等待

int pthread_join(pthread_t thread, void **value_ptr);

第一个参数是线程标识,第二是传出型参数用来获取线程的返回值

注意:

  • 调用该函数的线程将挂起等待,直到id为thread的线程终止
  • 这里获取的线程退出信息并没有终止信号信息,而终止信号信息是对于整个进程来说的,如果线程收到信号崩溃也会导致整个进程也崩溃
  • thread线程以不同的方法终止,通过pthread_join得到的终止状态是不同的

终止获取的状态情况:

  • 如果thread线程通过return返回,value_ ptr所指向的单元里存放的是thread线程函数的返回值
  • 如果thread线程被别的线程调用pthread_ cancel异常终掉,value_ ptr所指向的单元里存放的是常数PTHREAD_ CANCELED
  • 如果thread线程是自己调用pthread_exit终止的,value_ptr所指向的单元存放的是传给pthread_exit的参数
  • 如果对thread线程的终止状态不感兴趣,可以传NULL给value_ ptr参数

线程分离

int pthread_detach(pthread_t thread);

默认情况下,新创建的线程是joinable的,线程退出后,需要对其进行pthread_join操作,否则无法释放资源,从而造成系统泄漏

如果不关心线程的返回值,join是一种负担,这个时候,我们可以告诉系统,当线程退出时,自动释放线程资源

注:

可以是线程组内其他线程对目标线程进行分离,也可以是线程自己分离:

pthread_detach(pthread_self());

joinable和分离是冲突的,一个线程不能既是joinable又是分离的

线程的分离也是具有一定延时性,分离之后如果再进行等待那么得到返回的结果是未定义的

线程分离后只是回收的时候自动进行回收,如果主线程先退出,那么整个进程也会退出;如果分离的线程执行崩溃,同样的整个进行也会崩溃

到此这篇关于C语言由浅入深讲解线程的定义的文章就介绍到这了,更多相关C语言线程内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C语言创建线程thread_create()的方法

    在头文件 threads.h 中,定义和声明了支持多线程的宏.类型和函数.所有直接与线程相关的标识符,均以前缀 thrd_ 作为开头.例如,thrd_t 是一个对象类型,它标识了一个线程. 函数 thrd_create()用于创建并开始执行一个新线程.函数 thrd_create()的其中一个参数为在新线程中需要被执行的函数 thrd_create()的其中一个参数为在新线程中需要被执行的函数.thrd_create()的完整原型是: int thrd_create(thrd_t *thr, t

  • c语言多线程编程使用示例

    复制代码 代码如下: #include <stdio.h>#include <stdlib.h>#include <pthread.h> #define THREAD_NUM 10 void *test(void *args) { printf("tid %d: i say 'Hello'.\n", args); return NULL;} int main() { int i, err; pthread_t child[THREAD_NUM]; f

  • 详解C语言编程之thread多线程

    目录 线程创建与结束 线程的创建方式: 线程的结束方式: join() detach() 互斥锁 <mutex> 头文件介绍 std::mutex 介绍 std::lock_guard std::unique_lock 示例: 原子变量 线程同步通信 线程死锁 死锁概述 死锁产生的条件 示例: 总结 线程创建与结束 C++11 新标准中引入了四个头文件来支持多线程编程,他们分别是<atomic> ,<thread>,<mutex>,<condition

  • C语言由浅入深讲解线程的定义

    目录 线程的概念 线程的创建 线程的终止 线程标识的比较 线程的取消 线程等待 线程分离 线程的概念 可以简单理解为一个正在独立运行的函数 注: 1.posix线程是一套标准吗,而不是实现 2.线程标识: phread_t,可能是整形也可能是结构体指针等 *简单介绍关于线程标识的函数* *pthread_equarl() ;判断两个线程标识是否相等**pthread_self():返回自身的线程标识* 线程的创建 pthread_creat(); int pthread_create( pthr

  • C语言由浅入深讲解文件的操作上篇

    目录 为什么使用文件 什么是文件 文件名 关于文件的一些概念 文件函数 fopen fclose 实例代码 绝对路径 文件的打开方式 文件操作流程 为什么使用文件 前面写的通讯录,增加人数退出程序后,数据就会消失.此时数据是存放在内存中,下次运行通讯录程序的时候,数据又得重新录入,如果使用这样的通讯录就很难受. 所以文件操作就应运而生.数据持久化的方法有两种:1.把数据存放在磁盘文件2.存放到数据库使用文件我们们可以将数据直接存放在电脑的硬盘上,做到了数据的持久化. 什么是文件 但是在程序设计中

  • C语言细致讲解线程同步的集中方式

    目录 互斥锁 条件变量 信号量 读写锁 互斥锁 使用互斥量完成对临界区的资源的加锁操作,使得同一时刻,对一个共享数据的使用只能又一个线程完成 例向屏幕上一次打印abcd四个字母 可以使用的是一个类似锁连的思想 a 加完解开后拿b锁依次类推 #define THRNUM 4 static pthread_mutex_t mut[4]; static int next(int n) { if(n + 1 == THRNUM) return 0; return n+1; } static void*

  • C语言由浅入深讲解文件的操作下篇

    目录 文件的顺序读写 字符输入输出fgetc和fputc 文本行输入输出函数fgets和fputs 格式化输入输出函数fscanf和fprintf 二进制输入输出函数fread和fwrite 文件的随机读写 fseek ftell fwind 文本文件和二进制文件 文件结束的判定 feof 文件缓冲区 第一篇讲了文件的基本概念,和文件如何打开和关闭.第二篇主要介绍文件的顺序读写和随机读写.外加文件缓冲区的知识点. 文件的顺序读写 字符输入输出fgetc和fputc fgetc:字符输入函数,也就

  • C语言深入讲解宏的定义与使用方法

    目录 一.C语言中的宏定义 二.宏定义表达式 三.宏表达式与函数的对比 四.有趣的问题 五.强大的内置宏 六.小结 一.C语言中的宏定义 #define是预处理器处理的单元实体之一 #define 定义的宏可以出现在程序的任意位置 #define 定义之后的代码都可以使用这个宏 #define 定义的宏常量可以直接使用 #define 定义的宏常量本质为字面量 下面的宏常量定义正确吗? 编写代码来测试: #define ERROR -1 #define PATH1 "D:\test\test.c

  • 由浅入深讲解Javascript继承机制与simple-inheritance源码分析

    老生常谈的问题,大部分人也不一定可以系统的理解.Javascript语言对继承实现的并不好,需要工程师自己去实现一套完整的继承机制.下面我们由浅入深的系统掌握使用javascript继承的技巧. 1. 直接使用原型链 这是最简粗暴的一种方式,基本没法用于具体的项目中.一个简单的demo如下: function SuperType(){ this.property = true; } SuperType.prototype.getSuperValue = function(){ return th

  • Java 超详细讲解类的定义方式和对象的实例化

    目录 1.面对对象的初步认识 1.1什么是面向对象 1.2面向对象与面向过程 2.类的定义与使用 2.1简单认识类 2.2 类的定义格式 3.类的实例化 3.1什么是实例化? 3.2重点笔记 总结 1.面对对象的初步认识 1.1什么是面向对象 用面向对象的思想来涉及程序,更符合人们对事物的认知,对于大型程序的设计.扩展以及维护都非常友好. 1.2面向对象与面向过程 举一个买手机的例子 以面向对象的方式来处理买手机这件事的话,我们就不需要关注买手机的过程,具体手机怎么买,如何到手,用户不用去关心,

  • C语言深入讲解函数的使用

    目录 关于函数 1. 函数的定义形式 2. 函数的声明 3. 返回语句 4. 函数参数 4.1 形式参数(传值调用) 4.2 实际参数(传址调用) 4.3 无参数 5. 函数的调用 5.1 嵌套调用 5.2 函数递归 总结: 关于函数 函数是C语言的基本单元,函数中包含实现程序功能的代码. C语言程序的入口位于main()函数之中,为了方便组织配合,调试和运行,一般是用main函数作为主调函数,通过调用其他的函数来实现程序的运行. 他们的关系图如下: 1. 函数的定义形式 先给大家看看完整的函数

  • C语言全方位讲解数组的使用

    目录 一维数组的创建和初始化 1.数组的创建 2.数组创建方式 3.数组的初始化 一维数组的使用 一维数组的存储 二维数组的创建与初始化 1.二维数组的创建 2.二维数组的初始化 二维数组的存储 数组的越界 总结 接着上次的操作符的详解,让我们来简单了解C语言里的数组. 一维数组的创建和初始化 1.数组的创建 数组是一组相同类型的元素的集合. 2.数组创建方式 type_t(数组类型) arr_name(数组名) [const_n](用来指定数组大小) 3.数组的初始化 数组的初始化是在其定义的

  • C语言图文并茂讲解分支语句用法

    目录 一.if 语句分析 二.switch 语句分析 三.小结 一.if 语句分析 if 语句用于根据条件选择执行语句 else 不能独立存在且总是与它最近的 if 相匹配 else 语句后可以接连其他 if 语句 if 语句中零值比较的注意点 bool 型变量应该直接出现于条件中,不要进行比较 变量和 0 值比较时,0 值应该出现在比较符号左边(这条规则可以拓展为任意字面量与变量比较时,字面量应该放在左边,变量放在右边,这样即使手误写成了 = ,编译器也能发现) float 型变量不能直接进行

随机推荐