线程知识点

一、线程

1.定义

线程:是一个进程并发执行多种任务的机制。

串行:多个任务有序执行,一个任务执行完毕后,再去执行下一个任务

并发:多个任务在单个CPU上运行,同一个时间片上只能运行一个任务,cpu不停在各个任务上切换

并行:多任务在多个cpu上运行,同一个时间片上可以执行多个任务

其中上下文:运行一个进程所需要的所有资源

上下文切换:切换进程时,cpu访问的资源需要替换原先的资源,进程的上下文切换是个耗时操作,所以引入线程。

因为线程属于同一进程下,共享其附属进程的所有资源。

2.进程和线程的区别

1.进程时资源分配的最小单位,线程是任务运行的最小单位

2.进程和进程之间相互独立,内核空间共享。进程之间数据通信需要引进IPC通信机制

3.线程与线程之间共享其附属进程的所有资源,所以线程之间通信不需要通信机制,但是需要注意同步互斥

4.多线程的效率比多进程高,多进程的稳定性比多线程高,多进程的资源量比多线程高

二、线程的创建  pthread_creat

gcc时需要加-pthread

#include <head.h>// 线程执行体,里面写副线程要执行的内容
void *callBack(void *arg) // void *arg=NULL
{while (1){printf("副线程\n");sleep(1);}return NULL;
}int main(int argc, char const *argv[])
{// 创建一个分支线程pthread_t tid;if (pthread_create(&tid, NULL, callBack, NULL) != 0){// 第一个参数:线程成功创建后的tid号// 第二个参数:线程属性,一般填NULL,代表默认属性// 第三个参数:回调函数,void *(callback)(void *) 函数指针,指向返回值void*类型,参数列表式void*类型的函数// 第四个参数:传递给回调函数的参数fprintf(stderr, "创建线程失败");}while (1){printf("主线程\n");sleep(1);}return 0;
}

 

pthread_creat传参

i.主线程向子线程传参

#include <head.h>// 线程执行体,里面写副线程要执行的内容
void *callBack(void *arg) // void *arg=NULL
{while (1){printf("副线程 a=%d,&a=%p\n", *(int *)arg, arg);sleep(1);}return NULL;
}int main(int argc, char const *argv[])
{int a = 10;// 创建一个分支线程pthread_t tid;if (pthread_create(&tid, NULL, callBack, (void *)&a) != 0){fprintf(stderr, "创建线程失败");}while (1){printf("主线程   a=%d,&a=%p\n", a, &a);sleep(1);}return 0;
}

ii.子线程向主线程传参

#include <head.h>// 线程执行体,里面写副线程要执行的内容
void *callBack(void *arg) // void *arg=&pa   &pa本身是int **类型
{int a = 10;*(int **)arg = &a; // 二级指针int**类型,解引用后,访问的是int**类型// 如果不强转,void*类型,就不知道要访问多少个字节while (1){printf("副线程 a=%d,&a=%p\n", a, &a);sleep(1);}return NULL;
}int main(int argc, char const *argv[])
{int *pa = NULL;// 创建一个分支线程pthread_t tid;if (pthread_create(&tid, NULL, callBack, (void *)&pa) != 0) // 要修改pa的值就要把pa的地址传过去{fprintf(stderr, "创建线程失败");}while (1){if (pa != NULL){printf("主线程   a=%d,&a=%p\n", *pa, pa);sleep(1);}}return 0;
}

三、线程的退出与回收 pthread_exit和pthread_join

i.不接收退出状态值

#include <head.h>// 线程执行体,里面写副线程要执行的内容
void *callBack(void *arg) // void *arg=NULL
{int i = 0;while (i < 3){printf("分支线程\n");sleep(1);i++;}printf("分支线程准备退出\n");pthread_exit(NULL); // 用于退出线程,不想传递退出状态值就填NULLreturn NULL;
}int main(int argc, char const *argv[])
{// 创建一个分支线程pthread_t tid;if (pthread_create(&tid, NULL, callBack, NULL) != 0){fprintf(stderr, "创建线程失败");}pthread_join(tid, NULL);//阻塞等待分支进程结束// 第一个参数:分支线程的tid号// 第二个参数:null为不接收线程退出的状态值// 不为null则将pthread_exit传递的退出状态复制到该二级指针指向的一级指针中。printf("主线程准备退出\n");return 0;
}

ii.接收退出的状态值 

#include <head.h>// 线程执行体,里面写副线程要执行的内容
void *callBack(void *arg) // void *arg=NULL
{int i = 0;while (i < 3){printf("分支线程\n");sleep(1);i++;}printf("分支线程准备退出\n");static int a = 10; // 若不加static,则分支进程结束了之后,a也将不存在,加static来延长生命周期pthread_exit(&a);return NULL;
}int main(int argc, char const *argv[])
{// 创建一个分支线程pthread_t tid;if (pthread_create(&tid, NULL, callBack, NULL) != 0){fprintf(stderr, "创建线程失败");}void *ptr = NULL;pthread_join(tid, &ptr);printf("%d\n", *(int *)ptr);printf("主线程准备退出\n");return 0;
}

练习:分别用两个线程来拷贝一张图片的前半部分和后半部分

#include <head.h>// 拷贝前半部分
void *callback1(void *arg)
{// 以读的方式打开文件int fd_r = open("./1.png", O_RDONLY);if (fd_r < 0){perror("open");return NULL;}// 以写的方式打开文件int fd_w = open("./copy.png", O_WRONLY);if (fd_w < 0){perror("open");return NULL;}off_t size = lseek(fd_r, 0, SEEK_END); // 通过文件的偏移量来计算文件的大小// lseek的返回值为距离文件开头的偏移量// 修改文件偏移量到文件开头位置lseek(fd_r, 0, SEEK_SET);lseek(fd_w, 0, SEEK_SET);char c = 0;for (int i = 0; i < size / 2; i++){read(fd_r, &c, 1);write(fd_w, &c, 1);}printf("前半部分拷贝完\n");// 关闭文件close(fd_r);close(fd_w);pthread_exit(NULL);
}// 拷贝后半部分
void *callback2(void *arg)
{// 以读的方式打开文件int fd_r = open("./1.png", O_RDONLY);if (fd_r < 0){perror("open");return NULL;}// 以写的方式打开文件int fd_w = open("./copy.png", O_WRONLY);if (fd_w < 0){perror("open");return NULL;}off_t size = lseek(fd_r, 0, SEEK_END); // 通过文件的偏移量来计算文件的大小// lseek的返回值为距离文件开头的偏移量// 修改文件偏移量到文件开头位置lseek(fd_r, size / 2, SEEK_SET);lseek(fd_w, size / 2, SEEK_SET);char c = 0;for (int i = size / 2; i < size; i++){read(fd_r, &c, 1);write(fd_w, &c, 1);}printf("后半部分拷贝完\n");// 关闭文件close(fd_r);close(fd_w);pthread_exit(NULL);
}
int main(int argc, char const *argv[])
{// 两个线程在拷贝前,确保文件存在且是清空状态int fd_w = open("./copy.png", O_WRONLY | O_CREAT | O_TRUNC, 0664);if (fd_w < 0){perror("open");return -1;}close(fd_w);// 创建两个线程pthread_t tid1, tid2;if (pthread_create(&tid1, NULL, callback1, NULL) != 0){fprintf(stderr, "分支线程1创建失败\n");return -1;}if (pthread_create(&tid2, NULL, callback2, NULL) != 0){fprintf(stderr, "分支线程1创建失败\n");return -1;}// 阻塞等待分支线程退出pthread_join(tid1, NULL);pthread_join(tid2, NULL);return 0;
}

四、分离线程 pthread_detach

分离线程,线程退出后资源由内核自动回收

当使用pthread_detach分离tid后,pthread_join就无法再回首tid线程的资源了且pthread_join不再阻塞

所以pthread_join和pthread_detach挑一个使用。

#include <head.h>// 线程执行体,里面写副线程要执行的内容
void *callBack(void *arg) // void *arg=NULL
{printf("副线程\n");pthread_exit(NULL);return NULL;
}int main(int argc, char const *argv[])
{// 创建一个分支线程pthread_t tid;if (pthread_create(&tid, NULL, callBack, NULL) != 0){fprintf(stderr, "创建线程失败");}pthread_detach(tid);printf("主线程\n");return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/690762.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

BIERv6测试解析— 如何使用仪表进行转发性能测试

什么是BIERv6 BIERv6&#xff08;Bit Index Explicit Replication IPv6 encapsulation&#xff09;是一种新型组播方案。 BIERv6使用比特串封装目的节点集合&#xff0c;无需建立组播分发树或保存流状态&#xff0c;简化了网络节点操作。它与SRv6无缝融合&#xff0c;高效承载…

Linux(Ubuntu24.04) 安装 MinIO

本文所使用的 Ubuntu 系统版本是 Ubuntu 24.04 ! # 1、下载 MinIO wget https://dl.min.io/server/minio/release/linux-amd64/minio# 2、添加可执行权限 chmod x minio# 3、导出环境变量&#xff0c;用于设置账号密码&#xff0c;我设置的账号和密码都是 minioadmin export MI…

霍金《时间简史 A Brief History of Time》书后索引(E--H)

A–D部分见&#xff1a;霍金《时间简史 A Brief History of Time》书后索引&#xff08;A–D&#xff09; 图源&#xff1a;Wikipedia INDEX E Earth: circumference, motion, shape Eclipses Eddington, Arthur Einstein, Albert: biography, see also Relativity; Special…

主机和ubuntu连接

在这里插入图片描述 提示&#xff1a;文章 文章目录 前言一、背景二、 2.1 2.2 总结 前言 前期疑问&#xff1a; 本文目标&#xff1a; 一、背景 最近在hw使用Clion连接服务器&#xff0c;就想把自己的电脑配置好&#xff0c;翻出来正点原子的教程&#xff0c;【正点原子】…

动手学深度学习16 Pytorch神经网络基础

动手学深度学习16 Pytorch神经网络基础 1. 模型构造2. 参数管理1. state_dict()2. normal_() zeros_()3. xavier初始化共享参数的好处 3. 自定义层4. 读写文件net.eval() 评估模式 QA 1. 模型构造 定义隐藏层–模型结构定义前向函数–模型结构的调用 import torch from torch…

正点原子FreeRTOS学习笔记——列表与列表项

目录 一、什么是列表和列表项 1、概念 2、FreeRTOS代码 &#xff08;1&#xff09;列表 &#xff08;2&#xff09;列表项 &#xff08;3&#xff09;迷你列表项 二、列表与列表项初始化 1、列表初始化 2、列表项初始化 三、列表插入与删除列表项 1、原理解释 2、…

HDFS- DataNode磁盘扩缩容

HDFS- DataNode磁盘扩缩容 背景: 缩减/增加节点磁盘 方案介绍: 采用hdfs dfsadmin -reconfig 动态刷新配置实现,不停服扩缩容。 注意事项: 请在进行缩容之前,务必了解实际的数据量,并确保磁盘有足够的空间来容纳这些数据。还需要考虑未来的使用需求,要预留一定数量的空间…

数据结构之——队列详解

目录 前言&#xff1a; 一、什么是队列 二、队列的实现 2.1 队列结构 2.2 队列初始化 2.3 队列销毁 2.4 入队列 2.5 出队列 2.6 获取队列头部元素 2.7 获取队列尾部元素 2.8 获取队列中有效元素个数 2.9 检测队列是否为空 三、 代码总览 Queue.h test.c 四、例题 前言…

JVM---垃圾回收

目录 一、C/C的内存管理 二、Java的内存管理 三、垃圾回收的对比 四、Java的内存管理和自动垃圾回收 五、方法区的回收 手动触发回收 六、堆回收 如何判断堆上的对象可以回收&#xff1f; 引用计数法 可达性分析算法 五种对象引用 软引用 软引用的使用场景-缓存 弱引用 虚…

BGP(border gateway protocol)边界网关协议初识篇

BGP它是一种路径矢量协议&#xff0c;用于决定数据包在互联网中的最佳路径。 1、工作原理&#xff1a; 自治系统&#xff08;AS&#xff09;间路由: BGP主要用于连接不同自治系统之间的路由器&#xff0c;其中每个自治系统&#xff08;AS&#xff09;代表一组具有共同路由的网…

动态规划解决回文子串问题

前言&#xff1a; 回文串相关问题在我们的算法题中算是老生常谈&#xff0c;本文主要介绍如何使用动态规划的思路去解决回文串系列问题。 总体思路&#xff1a; 能够将所有的子串是否是回文的信息&#xff0c;存储在二维dp表中。有了这个dp表&#xff0c;就可以将hard难度转…

LeetCode - 0088 合并两个有序数组

题目地址&#xff1a;https://leetcode.cn/problems/merge-sorted-array/description/ 引言&#xff1a;话接上回&#xff0c;由于上次面试官着急下班&#xff0c;面试不得不提前终止&#xff0c;这不&#xff0c;他又找我去面试了 面试官&#xff1a;你好&#xff0c;小伙子&a…