linux 线程

文章目录

    • 1.线程概念
      • 线程优点
      • 线程缺点
    • 2.Linux线程和进程
    • 3. Linux线程控制
      • 线程创建---pthread_create
      • 线程等待---pthread_join
      • 线程退出
      • 线程分离---pthread_detach

1.线程概念

进程=内核数据结构 + 进程代码和数据
我们的代码在进程中,全部都是串行调用的
进程创建,成本较高(时间和空间),不仅需要创建PCB,页表等,还需要进行初始化,时间和空间成本高,所以多进程的目的是为了多执行流并发执行
地址空间和地址空间上的虚拟地址,本质是一种资源,一个进程的大部分资源都可以通过地址空间来访问
在内核角度,进程是承担分配系统资源的基本实体

当一个程序被执行时,操作系统会为其创建一个进程,并为该进程分配必要的系统资源,如内存空间、设备I/O等。这些资源是进程执行程序所必需的,只有被分配给进程后,才能被进程所使用。

加粗样式
线程是进程内部的一个执行分支,是CPU调度的基本单位,线程在进程地址空间内运行
Linux设计者认为,进程和线程都是执行流,具有极度的相似性,没必要单独设计数据结构和算法,直接复用代码,用进程模拟线程,也就是说,Linux中没有真正的线程
CPU调度时不需要区分进程和线程,都是执行流,在Linux中,所有的调度执行流都叫做轻量级进程(LWP),OS在调度时,使用LWP来进行调度
可以用是 ps -aL 查看轻量级进程
在这里插入图片描述

线程优点

创建一个新线程的代价比创建一个新进程的代价小得多
与进程切换相比,线程之间的切换需要OS做的工作要少很多—CPU内存在cache,切换进程,需要将CPU内的硬件级cache数据丢弃,重新填充
线程占用的资源比进程要少很多

线程缺点

健壮性降低:健壮性主要指的是线程在并发执行过程中,面对各种问题时,能够保持其正常功能和性能的能力,编写多线程程序需要更深入的考虑
缺乏访问控制:进程是访问控制的基本粒度,在一个线程中调用某些OS函数会对整个进程造成影响。
编程难度提高:编写与调试一个多线程程序比单线程程序困难得多

2.Linux线程和进程

  • 进程是承担分配系统资源的基本实体
  • 线程是CPU调度的基本单位
  • 线程私有的数据:
    调度:CPU运行的上下文数据
    常规运行:独立的栈结构
  • 线程在进程地址空间内运行,在同一进程的同一地址空间中,所有线程的数据区(已初始化数据区和未初始化数据区)和代码区都是共享的,定义的全局变量每个线程都可以访问到

3. Linux线程控制

  • Linux中没有真正的线程,只有轻量级进程,所以Linux系统不会有线程相关的系统调用,只有轻量级进程的系统调用
  • 为了用户更好使用,有了pthread库(原生线程库)),将轻量级进程系统调用进行封装,转换为线程相关接口提供给用户。不属于内核,用户级
  • 要使用这些接口,需要包含头文件<pthread.h>,同时编译时指定库 -lpthread

线程创建—pthread_create

在这里插入图片描述
thread:输出型参数,返回线程ID
attr:设置线程的属性,默认设置为NULL即可
start_routine:参数为void*,返回值为void*的函数指针,是线程启动后执行的函数
arg:传给线程启动函数的参数
在这里插入图片描述
pthreads函数成功返回0,失败返回错误码

#include <iostream>
#include <unistd.h>
#include <pthread.h>void* threadrun(void*)
{while(true){std::cout << "I am new thread" << std::endl;sleep(1);}
}int main()
{pthread_t tid;pthread_create(&tid, nullptr, threadrun, nullptr);while(true){std::cout << "I am main thread" << std::endl;sleep(1);}return 0;
}

在这里插入图片描述

  • 获取线程ID pthread_self
  • 获取线程ID: 1.创建线程时输出型参数获取 2.pthread_self
#include <iostream>
#include <string>
#include <unistd.h>
#include <pthread.h>std::string ToHEX(pthread_t tid)
{char buff[128];snprintf(buff, sizeof(buff) - 1, "%#lx", tid);return buff;
}void* threadrun(void*)
{while(true){std::cout << "I am new thread, tid :" << ToHEX(pthread_self()) << std::endl;sleep(1);}
}int main()
{pthread_t tid;pthread_create(&tid, nullptr, threadrun, nullptr);while(true){std::cout << "I am main thread, new thread tid:" << ToHEX(tid) << std::endl;sleep(1);}return 0;
}

在这里插入图片描述

线程等待—pthread_join

线程被创建,也需要被等待,不等待会有内存泄露的问题,同时也需要获取线程的退出信息
在这里插入图片描述
retval:输出型参数,为线程函数的返回值

#include <iostream>
#include <string>
#include <vector>
#include <unistd.h>
#include <pthread.h>std::string ToHEX(pthread_t tid)
{char buff[128];snprintf(buff, sizeof(buff) - 1, "%#lx", tid);return buff;
}struct Thread
{Thread(pthread_t tid, const std::string& name):_tid(tid), _threadName(name){}pthread_t _tid;std::string _threadName;
};void* threadrun(void* args)
{std::string* name = static_cast<std::string*>(args);int cnt = 10;while(cnt--){std::cout << *name << " is running...." << std::endl;sleep(1);}return (void*)100;
}int main()
{const int threadNum = 5;std::vector<Thread> threads;for (int i = 0; i < threadNum; i++){pthread_t tid;std::string* name = new std::string("thread-" + std::to_string(i + 1));pthread_create(&tid, nullptr, threadrun, name);threads.emplace_back(tid, *name);}void* ret;for (auto& thread : threads){pthread_join(thread._tid, &ret);std::cout << thread._threadName << "is quit, return val: " << (long long int)ret << std::endl;}return 0;
}

在这里插入图片描述

线程退出

    1. 从线程函数直接return
    1. pthread_exit
      在这里插入图片描述
      retval:退出信息
#include <iostream>
#include <string>
#include <vector>
#include <unistd.h>
#include <pthread.h>void* threadrun(void*)
{int cnt = 5;while(cnt--){std::cout << "线程还有" << cnt << "秒退出" << std::endl;sleep(1);}std::string* ret = new std::string("11111111111111111111111111");pthread_exit(ret);
}int main()
{pthread_t tid;pthread_create(&tid, nullptr, threadrun, nullptr);void* ret;pthread_join(tid, &ret);std::string* retval = static_cast<std::string*>(ret);std::cout << "thread return val: " << *retval << std::endl;return 0;
}

在这里插入图片描述
线程调用pthread_exit()终止,pthread_join()所获得的返回值为pthread_exit()的参数

    1. 一个线程调用pthread_cancel终止同一进程中的另一个线程保证新线程已经启动的前提下
      在这里插入图片描述
#include <iostream>
#include <string>
#include <vector>
#include <unistd.h>
#include <pthread.h>void* threadrun(void*)
{while(true){std::cout << "new thread running...." << std::endl;sleep(1);}
}int main()
{pthread_t tid;pthread_create(&tid, nullptr, threadrun, nullptr);sleep(2);pthread_cancel(tid);void* ret;pthread_join(tid, &ret);std::cout << "thread return val: " << (long long int)ret << std::endl;return 0;
}

在这里插入图片描述
线程被取消,退出信息是-1,-1是一个宏
在这里插入图片描述

线程分离—pthread_detach

在这里插入图片描述

  • 默认情况下,线程是需要被等待的,线程退出后,要对其进行pthread_join操作,否则无法释放资源,造成资源泄露
  • 如果不关心线程返回值,可以分离线程,线程退出时,自动释放资源
    分离是线程的一种工作状态,底层依旧是同一个进程,只是不需要且禁止join了

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

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

相关文章

入侵检测系统(IDS)

入侵检测 入侵检测&#xff08;Intrusion Detection&#xff09;是指发现或确定入侵行为存在或出现的动作&#xff0c;也就是发现、跟踪并记录计算机系统或计算机网络中的非授权行为&#xff0c;或发现并调查系统中可能为视图入侵或病毒感染所带来的异常活动。 入侵检测系统 …

upload-labs第十三关教程

upload-labs第十三关教程 第十三关一、源代码分析代码审计 二、绕过分析1&#xff09;0x00绕过a.上传eval.pngb.使用burpsuite进行拦截修改之前&#xff1a;修改之后&#xff1a;进入hex模块&#xff1a; c.放包上传成功&#xff1a; d.使用中国蚁剑进行连接 2&#xff09;%00绕…

20240619在飞凌OK3588-C的LINUX系统启动的时候拉高3个GPIO口141-111-120【方法一】

20240619在飞凌OK3588-C的LINUX系统启动的时候拉高3个GPIO口141-111-120【方法一】 2024/6/19 16:12 缘起&#xff1a;在凌OK3588-C的LINUX R4系统启动的时候&#xff0c;需要拉高GPIO4_B5、GPIO3_B7和GPIO3_D0。 修改rcS&#xff0c;在系统启动的时候&#xff0c;即可拉高。 通…

极具吸引力的小程序 UI 风格

极具吸引力的小程序 UI 风格

002.Linux CentOS7 安装

我 的 个 人 主 页&#xff1a;&#x1f449;&#x1f449; 失心疯的个人主页 &#x1f448;&#x1f448; 入 门 教 程 推 荐 &#xff1a;&#x1f449;&#x1f449; Python零基础入门教程合集 &#x1f448;&#x1f448; 虚 拟 环 境 搭 建 &#xff1a;&#x1f449;&…

湖北文理学院2024年成人高等继续教育招生简章

湖北文理学院&#xff0c;作为一所历史悠久、底蕴深厚的学府&#xff0c;始终致力于为社会各界培养具备高素质、专业技能和创新精神的优秀人才。在成人高等继续教育领域&#xff0c;湖北文理学院更是凭借其卓越的教学质量和丰富的教育资源&#xff0c;吸引了众多有志于提升自身…

ThinkPHP5大学生社会实践管理系统

有需要请加文章底部Q哦 可远程调试 ThinkPHP5大学生社会实践管理系统 一 介绍 大学生社会实践管理系统基于ThinkPHP5框架开发&#xff0c;数据库mysql&#xff0c;前端bootstrap。系统角色分为用户和管理员。 技术栈&#xff1a;ThinkPHP5mysqlbootstrapphpstudyvscode 二 功…

Gobject tutorial 七

The GObject base class GObject是一个fundamental classed instantiatable type,它的功能如下&#xff1a; 内存管理构建/销毁实例set/get属性方法信号 /*** GObjectClass:* g_type_class: the parent class* constructor: the constructor function is called by g_object…

基于单片机的智能窗户控制系统的设计

摘 要&#xff1a; 根据单片机技术和现代传感器技术 &#xff0c; 本文主要针对基于单片机的智能窗户控制系统的设计进行探讨 &#xff0c; 仅供参考 。 关键词&#xff1a; 单片机 &#xff1b; 智能窗户 &#xff1b; 控制系统 &#xff1b; 设计 在现代科学技术持续发展的带…

【精品方案】产业园区数字孪生规划方案(39页PPT)

引言&#xff1a;随着数字化和智能化技术的快速发展&#xff0c;传统产业园区面临着转型升级的重大机遇。数字孪生技术作为一种将物理世界与数字世界紧密结合的创新技术&#xff0c;为产业园区的规划、建设和运营管理提供了全新的解决方案。本方案旨在通过构建产业园区数字孪生…

Upload-Labs:Pass - 1(JS前端白名单)

Pass_1 1. 上传测试2. 代码审计**获取文件输入的值**&#xff1a;**检查是否选择了文件**&#xff1a;**定义允许的文件类型**&#xff1a;**提取文件的扩展名**&#xff1a;**检查文件类型是否允许上传**&#xff1a;**构建错误消息并提醒用户**&#xff1a; 3.绕过思路3.1 将…

集合系列(二十六) -利用LinkedHashMap实现一个LRU缓存

一、什么是 LRU LRU是 Least Recently Used 的缩写&#xff0c;即最近最少使用&#xff0c;是一种常用的页面置换算法&#xff0c;选择最近最久未使用的页面予以淘汰。 简单的说就是&#xff0c;对于一组数据&#xff0c;例如&#xff1a;int[] a {1,2,3,4,5,6}&#xff0c;…

一文带你读懂向量数据库(上)

大数据产业创新服务媒体 ——聚焦数据 改变商业 什么是向量数据库&#xff1f; 向量数据库的概述&#xff1a;向量数据库是一种数据库&#xff0c;专门设计用于存储和查询向量数据&#xff0c;常用于机器学习和数据科学领域。向量数据库可以高效地存储大规模的向量数据&#x…

STM32HAL库--NVIC和EXTI

1. 外部中断实验 1.1 NVIC和EXTI简介 1.1.1 NVIC简介 NVIC 即嵌套向量中断控制器&#xff0c;全称 Nested vectored interrupt controller。是ARM Cortex-M处理器中用于管理中断的重要组件。负责处理中断请求&#xff0c;分配优先级&#xff0c;并协调中断的触发和响应。 它是…

【千帆AppBuilder】你有一封邮件待查收|未来的我,你好吗?欢迎体验AI应用《未来信使》

我在百度智能云千帆AppBuilder开发了一款AI原生应用&#xff0c;快来使用吧&#xff01;「未来信使」&#xff1a;https://appbuilder.baidu.com/s/Q1VPg 目录 背景人工智能未来的信 未来信使功能介绍Prompt组件 千帆社区主要功能AppBuilderModelBuilder详细信息 推荐文章 未来…

Django REST framework数据展示技巧:分页、过滤与搜索的实用配置与实践

系列文章目录 Django入门全攻略&#xff1a;从零搭建你的第一个Web项目Django ORM入门指南&#xff1a;从概念到实践&#xff0c;掌握模型创建、迁移与视图操作Django ORM实战&#xff1a;模型字段与元选项配置&#xff0c;以及链式过滤与QF查询详解Django ORM深度游&#xff1…

k8s部署grafana beyla

k8s部署grafana beyla OS: Static hostname: test Icon name: computer-vm Chassis: vm Machine ID: 22349ac6f9ba406293d0541bcba7c05d Boot ID: 83bb7e5dbf27453c94ff9f1fe88d5f02 Virtualization: vmware Operating System: Ubuntu 22.04.4 LTS Kernel: Linux 5.15.0-105-g…

C#.Net筑基-类型系统①基础

C#.Net的BCL提供了丰富的类型&#xff0c;最基础的是值类型、引用类型&#xff0c;而他们的共同&#xff08;隐私&#xff09;祖先是 System.Object&#xff08;万物之源&#xff09;&#xff0c;所以任何类型都可以转换为Object。 01、数据类型汇总 C#.NET 类型结构总结如下图…

使用@Value注解无法成功获取配置文件内容,常见原因

在日常的java开发中&#xff0c;我们经常会遇到一些需要将信息写在配置文件的要求&#xff0c;比如文件的输出目录&#xff0c;输入目录的。当在配置文件中写入对应的目录配置时&#xff0c;那么怎么读取配置文件的内容就需要我们去了解了。 在java中一般使用Value这个注解去读…

SSM小区车辆信息管理系统-计算机毕业设计源码06111

摘 要 科技进步的飞速发展引起人们日常生活的巨大变化&#xff0c;电子信息技术的飞速发展使得电子信息技术的各个领域的应用水平得到普及和应用。信息时代的到来已成为不可阻挡的时尚潮流&#xff0c;人类发展的历史正进入一个新时代。在现实运用中&#xff0c;应用软件的工作…