C语言中的多线程简介

2023-05-16

线程 Thread

专业术语称之为程序执行流的最小单元 。线程是不会执行程序的,可以理解成线程就是一个载体,将 要执行的代码 运送到CPU进行处理。

多线程就是多个线程同时并发执行。

(注意并发与并行的区别,并行同时执行不同的任务,并行是交替执行不同的任务。)

1,为什么要用多线程?

1)避免阻塞

单个线程中的程序,是按照顺序执行的,排在前面的程序如果发生异常卡住(阻塞),会影响到后面的程序执行。多线程就等于是异步调用,避免这个情况。

2)避免CPU的空转

这个比如一个网页,如果是单线程的话,服务器处理一条请求后,会等待下一个请求,这时候CPU处于一个闲置的状态。多线程能避免这个问题。

3)提升效率

避免了1,2的问题,效率自然就提高了,归根结底也是为了这点。

2,线程与进程的区别

进程是系统进行资源分配和调度的一个独立单位,线程是进程的一个实体,是CPU调度和分派的基本单位,线程只是一段程序的执行。

比如打开了一个软件(比如说QQ),能从任务管理器,就能看到有QQ这样一个进程,这时候想跟别人聊个天,打开对话框,这就是运行一个线程;查看一下聊天的这个人的资料,这又运行了另外一个线程。 

同一个进程下的线程是资源共享的,进程与进程直间都是独立的。

写了几个简单程序来说明一下,怎么使用线程,线程中的问题。

---------------------------------------------------------------------------------------------------------------------------------------------------

  • 创建一个线程,看 example0.c
#include <stdio.h>
#include <stdlib.h>
#include <pthread.h> 
// 使用线程时需要添加<pthread.h>这个头文件
// myfunc 线程携带的函数;固定为Void*型
myfunc(void* args){
		printf("hello world\n");
	return NULL;
}

int main(){
    //  声明线程th1 
    pthread_t th1; 
    pthread_create(&th1,NULL,(void*)myfunc,NULL); 
    pthread_join(th1,NULL); 
}

C语言的程序中要使用线程,需要添加<pthread.h>这个头文件
myfunc()是这个线程要携带执行的程序 ,函数与函数的参数,都是必须是 void* 类 ,涉及到调用和传值时需要进行类型强转。

主函数中,先对线程进行一个声明, pthread_t th1;  使用 pthread_create()对th1进行创建。

pthread_create()中有4个参数

1要创建的线程id(th1);  2设置线程的属性(没有特殊需求时填NULL就可以了)  ; 3线程要运行的函数的地址(myfunc);4是要向运行的那个函数传参(myfunc()中的args),args 也是void*类型。

pthread_join()函数的功能是等待一个线程的结束,它是一个线程阻塞的函数。

pthread_join有两个参数:指定要等待的线程id;接收线程函数的返回值。

运行这个程序看看

如果没有pthread_join,终端可能会没有打印,因为主函数执行结束,线程中的函数还没有执行完成。

  • example1.c

一个多线程的例子,来演示说明下线程的是并发执行

// 创建两个线程  
void* myfunc(void* args){
	int i;
	char* name = (char*)args;
	for(i=1;i<50;i++){
		printf("%s:%d\n",name,i);
	}
	return NULL;
}

int main(){

	pthread_t th1;
	pthread_t th2;
	
	pthread_create(&th1,NULL,myfunc,"th1");
	pthread_create(&th2,NULL,myfunc,"th2");

	pthread_join(th1,NULL);
	pthread_join(th2,NULL);	

}

两个线程,都去调用这个myfunc()函数,运行结果如下

从这个程序的运行结果,能看出th1运行到17时,th2开始运行了,而这时th1没有继续打印,能看出th1与th2是并发执行的。两个线程的执行顺序是不确定的,重复运行example1.c这个程序的结果也会不同。

这个程序里使用到了pthread_create()中的第四个参数向 myfunc传参。

  • example3.c
int s =0;

void* myfunc(void* args){
	int i =0;
	for(i=0;i<10000;i++){
	s++;	
	}
	return NULL;
}
int main(){
	pthread_t th1;
	pthread_t th2;

	pthread_create(&th1,NULL,myfunc,NULL);
	pthread_create(&th2,NULL,myfunc,NULL);

	pthread_join(th1,NULL);
	pthread_join(th2,NULL);

	printf("s=%d\n",s);

	return 0;
}
// 用两个线程去执行myfunc函数 , 理想值应该为20000 

编译后运行结果如下

这个函数里用了一个全局变量s,执行函数是一个10000的累加,理想的运行结果应该是20000,我这里运行了3次这个程序,每次的结果都不同。 用这个演示来表示一下,多线程之间是资源共享的。 

s++ 是有三个步的,读取s,s+1,写入s。 

在程序运行的某个时刻,th1携带myfunc执行s++,读取s,此时s=100,进行s+1, 与此同时th2也开始读取s,此时的s还是等于100, 这时th1,执行写入s=101,th2执行s++,写入s ,s=101. th2中的s 就会覆盖掉 th1中的s 。这样造成了结果的误差。

    手绘感受一下,绝对记忆深刻。·

这个问题有的名字 叫 race conditon

那么怎么解决这样问题呢? 应该听过一个词, 锁 

  • example4.c

对example3.c中的线程进行一个加锁

int s =0;

pthread_mutex_t lock;   //定义一个锁

void* myfunc(void* args){
	pthread_mutex_lock(&lock); //上锁
	int i =0;
	for(i=0;i<10000;i++){
	s++;	
	}
	pthread_mutex_unlock(&lock);//解锁
	return NULL;
}
int main(){
	pthread_t th1;
	pthread_t th2;

	pthread_mutex_init(&lock,NULL); //初始化lock这个锁	
	
	pthread_create(&th1,NULL,myfunc,NULL);
	pthread_create(&th2,NULL,myfunc,NULL);

	pthread_join(th1,NULL);
	pthread_join(th2,NULL);

	printf("s=%d\n",s);

	return 0;
}

运行结果

锁的作用是什么呢? 

前面说过多线程是并发执行的,th1运行后 进行了加锁,th2这时候想要运行,就必须等待th1解锁之后才行。

(一个卫生间,多个人要用,第一个人进去之后,把门锁上了,后边的人就得排队等着,第一个方便完了,解锁开门出来,第二个人进去,继续锁门……)

锁 在提高程序的安全性的同时,也降低了程序的效率。

锁的使用方法

pthread_mutex_t lock;  声明一个锁

pthread_mutex_init(&lock,NULL); 对声明的锁进行初始化

pthread_mutex_lock(&lock); //上锁 此时其他线程就开始等待

pthread_mutex_lock(&unlock); //解锁  其他线程可以使用资源了

死锁!

拿上边举例,th1运行后,th2会等待th1解锁,才能运行,如果程序出现错误中断了,th1没有执行完,重新启动后,th1又重新执行,这时候th2排在th1前边,需要等待th1的解锁 ,而th1又在等待th2结束。 这样就造成了相互等待的情况,这个就是死锁。

额外

多线程的目的

可以充分利用CPU资源,并发去做很多事,提高使用率,减少计算时间。

线程是不是越多越好?

    肯定不是,要考虑线程的创建时间+销毁时间 ,选择一个适当的数量。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

C语言中的多线程简介 的相关文章

  • QT QCompleter 用法小结

    1 有何用处 QCompleter类提供基于项模型的补全功能 您可以使用QCompleter在任何Qt小部件 xff08 如QLineEdit和QComboBox xff09 中提供自动补全 当用户开始键入单词时 xff0c QComple
  • QT 关于全屏 与 退出全屏

    在某些场景 xff08 比如说 xff1a 车机系统 医疗仪器等 xff09 可能需要确保软件的全屏显示 xff0c 本次探讨的是如何实现全屏与退出全屏 实现全屏的几种方式 span class token operator lt span
  • QT 如何设置 QCompleter 样式

    在使用QCompleter xff0c 一般都会面临两个问题 xff1a 1 如何设置Completer的行高 xff1b 2 如何设置Completer的样式 本文将一次为你解答 xff0c 顺便讲讲ComboBox设置Completer
  • QT 如何知道用户点击 QTextBrowser 中的具体链接地址

    在通过鼠标点击 QTextBrowser 中的链接地址时 xff0c 如果是希望使用外部浏览器打开链接地址 xff0c 只需要添加一行代码 ui gt textBrowser gt setOpenExternalLinks true 但如果
  • QT 常用控件类型命名参考

    规范的定义 规范 xff0c 有名词 动词 代词等词性 意指符合逻辑 xff0c 客观 真实 全面 完整 准确 及时 xff0c 达标 明文规定或约定俗成的标准 如 xff1a 道德规范 技术规范等 或是指按照既定标准 规范的要求进行操作
  • C++ 如何让代码在main()函数之前或者之后执行

    程序最早执行的函数其实并不是main xff0c 在main函数之前会有一系列初始化的操作 xff08 通常是由链接器等完成的 xff09 比如在windows中 xff0c 这个函数是mainCRTStartup xff0c 这个函数是链
  • QT 如何提高 Qt Creator 的编译速度

    如何提高编译速度 xff0c 貌似是一个老生常谈的话题 对于Qter而言 xff0c 如何提高QT Creator 的编辑速度是一直都是大家所期盼的 本文也是查阅了各路大神的方法后整理出来的 xff0c 希望对各位有所帮助 1 在 pro文
  • 卧槽!出了一个Python实时目标跟踪系统神器!

    在当下自动驾驶 智慧城市 安防等领域对车辆 行人 飞行器等快速移动的物体进行实时跟踪及分析的需求可谓比比皆是 xff0c 但单纯的目标检测算法只能输出目标的定位 43 分类 xff0c 无法对移动的目标具体的运动行为及特征进行分析 xff0
  • C++ 什么是BCC校验

    BCC Block Check Character xff0c 俗称异或校验 BCC的实现方法 xff1a 将所有数据都和一个指定的初始值 通常是0 异或一次 xff0c 所得结果为校验值 BCC一般只是用来排错的 xff0c 并不是加密算
  • QT 如何让QLineEdit的文字从最左边开始显示

    当QLineEdit的文字长度超过了控件宽度 xff0c 一般而言 xff08 控件文字设为靠左对齐 xff09 xff0c 用户只能看到这一长串文字的后半段部分 但是如果想让QLineEdit从最左边开始显示字符串内容呢 xff1f 举个
  • QT setBuddy 用法小结

    将此标签的好友 xff08 窗口其他小部件 xff09 设置为好友 当用户按下此标签指示的快捷键时 xff0c 键盘焦点将转移到标签的好友小部件 好友机制仅适用于包含一个字符前缀为 amp 的文本的QLabels 此字符被设置为快捷键 设置
  • QT stackUnder 用法小结

    stackUnder是什么意思 xff1f 根据官方的解析 xff0c 简而言之一句话 xff1a 把窗口的小部件放置于父窗口的堆栈窗口中 本文将通过一则简单的示例 xff0c 来具体解析stackUnder该如何使用 官方解析 void
  • QT 线程安全的单例模式(使用Q_GLOBAL_STATIC实现)

    Qt提供了宏 Q GLOBAL STATIC xff08 官方说明文档 xff1a https doc qt io qt 5 qglobalstatic html xff09 xff0c 用于创建全局静态对象 鉴于此 xff0c 我们可以通
  • git push提示“fatal: the remote end hung up unexpectedly”的解决方法

    问题描述 在执行push命令时 xff0c 命令行提示 xff1a span class token function git span push origin master Enumerating objects 4968 done Co
  • 四旋翼无人机动力学模型及控制

    四旋翼无人机动力学模型及控制 I 欧拉角与旋转矩阵 Overview欧拉角与旋转矩阵Body Frame Angular Velocity and
  • Urbannav数据集/novatel_data/inspvax订阅

    使用Urbannav数据集做多原融合定位评估精度 xff0c 需要使用 novatel data inspvax作为真值 xff0c 但无法直接订阅该话题 rotopic echo novatel data inspvax 显示无法订阅 x
  • 软路由硬件, 研究了一圈还是J1900 , i211网卡又如何

    软路由的心路历程还真是纠结 xff0c 现写出来供大家参考 需求上也是几经翻车 xff1a 路由 xff0c NAS 之类的一体机是最先迸发出来也是最先被否的 本着大道至简 xff0c 设备专用的原则 软路由即路由 xff0c 不做他用 功
  • putty远程连接ubuntu18失败处理方法

    putty客户端远程连接报错 xff1a Network error Connection refused 解决方法 xff1a 1 先在Windows系统cmd命令下ping下此ip地址连接是否正常 xff0c 如果连接正常说明主机没有问
  • KITTI榜单新SOTA!相机-LiDAR双向融合新范式 | CVPR 2022 Oral & arXiv 2023

    点击下方卡片 xff0c 关注 CVer 公众号 AI CV重磅干货 xff0c 第一时间送达 点击进入 gt 计算机视觉 微信技术交流群 作者 xff1a 王利民 xff08 已授权转载 xff09 编辑 xff1a CVer https

随机推荐

  • C++ Primer 学习笔记 第七章 类

    233 定义类 span class token macro property span class token directive hash span span class token directive keyword include
  • 如何编写CMakeLists

    Preface 构建一个工程的时候 xff0c CMake需要知道的几个点 xff08 如果下面几件事你知道怎么做了 xff0c 多大的工程就都不是问题了 A 源代码在哪里 xff1f B 头文件在哪里 xff1f C 怎么生成静态或者动态
  • 【算法】最快最简单的排序——桶排序

    在我们生活的这个世界中到处都是被排序过的 站队的时候会按照身高排序 xff0c 考试的名次需要按照分数排序 xff0c 网上购物的时候会按照价格排序 xff0c 电子邮箱中的邮件按照时间排序 总之很多东西都需要排序 xff0c 可以说排序是
  • TCP、UDP数据包大小的限制

    1 概述 首先要看TCP IP协议 xff0c 涉及到四层 xff1a 链路层 xff0c 网络层 xff0c 传输层 xff0c 应用层 其中以太网 xff08 Ethernet xff09 的数据帧在链路层 IP包在网络层 TCP或UD
  • Jetson TX2刷机教程

    介绍 xff1a 本文介绍如何对Jetson TX2进行刷机 xff0c 系统版本为Jetpack4 6 0 准备 xff1a 主机 xff08 虚拟机 xff09 xff1a Ubuntu18 04Jetson TX2USB 连接线 刷机
  • 研扬Jetson NX镜像备份和恢复

    0 环境依赖 研扬RC S ARES 200AI NX CSC00型号Jetpack4 6 0 1 如何进入Recovery模式 1 先按住Recovery键再插电源上电 xff0c 此间一直按住Recovery键2 3秒之后松开 2 将U
  • SpringCloud-Eureka快速入门,集群搭建

    Eureka 个人主页 xff1a https blog csdn net hello list type 61 blog 前言 第一章 xff1a SpringCloud环境搭建 Rest使用 这里博主从说更新springcloud xf
  • SpringCloud-Ribbon和Feign快速上手

    Ribbon 个人中心 xff1a https blog csdn net hello list 前情提要 xff1a SpringCloud环境搭建 Rest使用SpringCloud Eureka快速入门 xff0c 集群搭建 首先我们
  • Java IO流详解

    Java IO流详解 个人主页 xff1a https blog csdn net hello list 今天我们来学习下java中的io部分 首先我们要知道io指的是什么 xff0c 输入输出 xff0c 就是输入输出流 xff0c 我们
  • SpringCloud-Hystrix服务熔断,快速入门

    Hystrix服务熔断 个人中心 xff1a 学习日记的博客 前情提要 xff1a SpringCloud环境搭建 Rest使用SpringCloud Eureka快速入门 xff0c 集群搭建SpringCloud Ribbon和Feig
  • 最新!!Intel首发UP SQUARED* GROVE物联网开发套件

    xfeff Intel官方网站正式推出UP SQUARED GROVE 物联网开发套件 https software intel com zh cn iot hardware up squared grove dev kit UP Squa
  • 再探指针:指针有什么用?(课堂笔记,来自翁恺老师的C语言进阶课,有着个人理解)

    指针的用处 xff1a 三个比较常用的场景 我们希望函数可以返回多个值的时候 我们都知道函数只可以return一个值 xff0c 使用指针便可以在一个函数里面返回多个值 举例 xff1a swap函数和数组中寻找最大值最小值的函数 xff0
  • java反射与注解详解,共同实现动态代理模式

    java反射与注解详解 xff0c 共同实现动态代理模式 个人主页 xff1a https blog csdn net hello list id xff1a 学习日记 不知不觉一年过去了 xff0c 整整一年 xff0c 这一年写了60多
  • 微信小程序快速入门

    微信小程序快速入门 在这里首先祝大家国庆节快乐 xff0c 其实原本文章都没有准备好 xff0c 也没有打算更文的 xff0c 那还是将就一下吧 xff0c 发个简单的 前言 相信大家对微信小程序并不陌生 xff0c 以前我们接触网络刚开始
  • RabbitMQ快速入门,这一篇看完教你学会

    RabbitMQ快速入门 今天学习RabbitMQ xff0c 你知道RabbitMQ是什么吗 xff0c RabbitMQ是一种消息中间件 xff0c 我们在写很多业务的时候 xff0c 有时候我们需要考虑到消息的实时性 xff0c 时效
  • 什么是协议栈

    协议栈是什么 1 协议栈是什么 简介 协议栈 xff0c 英语名称为Protocol stack xff0c 又称协议堆叠 xff0c 是计算机网络协议套件的一个具体的软件实现 协议套件中的一个协议通常是只为一个目的而设计的 xff0c 这
  • 任务,任务的切换,(TCB)

    任务也可以称作为进程 xff0c 是一个简单的程序 xff0c 该程序认为 CPU完全属于自己 xff0c 实时的应用的程序的设计的时候分割成了许多的任务 xff0c 每一个任务都对应应用的某一部分 每一个任务都被赋予一定的优先级 xff0
  • 浅谈pthread_setschedparam的使用

    浅谈pthread setschedparam 的使用 int pthread setschedparam pthread t target thread int policy const struct sched param param
  • 互斥量、临界区、信号量、事件标志组和消息邮箱

    http ejs90ejs iteye com blog 1351642 互斥量 临界区 信号量 事件标志组和消息邮箱 2010年07月23日 为了好的理解互斥量 临界区 信号量 事件标志组和消息邮箱 xff0c 下面一些知识对初学者来说很
  • C语言中的多线程简介

    线程 Thread 专业术语称之为程序执行流的最小单元 线程是不会执行程序的 xff0c 可以理解成线程就是一个载体 xff0c 将 要执行的代码 运送到CPU进行处理 多线程就是多个线程同时并发执行 xff08 注意并发与并行的区别 xf