线程与进程的深入解析及 Linux 线程编程

在操作系统中,进程和线程是进行并发执行的两种基本单位。理解它们的区别和各自的特点,能够帮助开发者更好地进行多任务编程,提高程序的并发性能。本文将探讨进程和线程的基础概念,及其在 Linux 系统中的实现方式,并介绍如何在 Linux 下进行线程编程。

进程与线程概念

进程

进程是程序的执行实例,是操作系统进行资源分配和调度的基本单位。每个进程都具有自己的独立内存空间、程序计数器、堆栈等资源。进程的创建过程是通过复制当前进程的数据来完成的,常见的系统调用有 fork() 和 vfork()

  1. fork():创建一个子进程,子进程是父进程的副本,拥有独立的内存空间。父进程和子进程的资源相互隔离。
  2. vfork():创建一个子进程时,父进程会被暂停,子进程共享父进程的内存空间,直到子进程调用 exec() 或 _exit() 来退出。

然而,进程的创建通常涉及大量的数据复制,影响系统的效率。这是因为每个进程都需要有独立的资源,造成一定的开销。

线程

线程是进程中的执行单位,也是操作系统调度的最小单位。与进程不同,线程不拥有独立的资源,而是与同一进程中的其他线程共享进程的资源(如内存、文件描述符等)。线程的创建和销毁比进程更加轻量级,因此线程通常用于提高并发性能。

一个线程通常由以下几个部分组成:

  • 线程ID (Thread ID):每个线程都有一个唯一的标识符。
  • 程序计数器:指向线程执行的下一条指令。
  • 寄存器:保存线程执行的中间状态。
  • :每个线程有自己的栈,用于保存局部变量和函数调用的上下文。

进程与线程的关系

  1. 线程依附于进程:每个线程都属于某个进程,不能脱离进程独立存在。
  2. 进程终止,线程随之终止:当一个进程终止时,其内部所有的线程都会被终止。
  3. 一个进程可以创建多个线程:这使得进程内可以同时执行多个任务,从而提高并发度。

进程与线程的区别

  • 资源分配

    • 进程是资源分配的单位,每个进程都有独立的资源。
    • 线程是调度执行的单位,多个线程共享同一进程的资源。
  • 创建开销

    • 进程创建时需要复制大量数据,开销较大。
    • 线程创建轻量级,开销较小。
  • 调度单位

    • 进程是操作系统调度的基本单位。
    • 线程是操作系统调度的最小单位。

Linux 下的线程编程

在 Linux 系统中,线程编程通常依赖于 NPTL(Native POSIX Thread Library),这是一种本地线程库,实现了 POSIX 标准中的线程相关接口。NPTL 库提供了线程的创建、执行、结束等功能。

创建线程

Linux 中可以通过 pthread_create 函数创建线程。该函数的原型如下:

int pthread_create(pthread_t *thread, const pthread_attr_t *attr,void *(*start_routine)(void *), void *arg);
  • thread:线程的标识符。
  • attr:线程属性,一般为 NULL 表示使用默认属性。
  • start_routine:线程执行的函数(即线程的入口函数)。
  • arg:传递给 start_routine 函数的参数。

线程的执行函数通常定义如下:

void *do_something(void *arg) {printf("Thread %d\n", getpid());return NULL;
}

创建线程后,主线程和新创建的线程将并发执行。

线程的结束

线程可以通过以下几种方式结束:

  1. 返回:线程执行函数返回时,线程结束,等价于调用 pthread_exit
  2. 调用 pthread_exit:主动调用 pthread_exit 结束线程。
  3. 线程被取消:通过 pthread_cancel 可以取消线程。
  4. 进程结束:如果进程结束,所有线程也会结束。
void pthread_exit(void *retval);
  • retval:线程退出时传递的返回值。

线程同步

线程共享进程的资源,这带来了数据竞争问题。为了避免多个线程同时访问共享数据导致的错误,我们通常使用互斥锁(mutex)等同步机制。

例如,可以使用 pthread_mutex_lock 和 pthread_mutex_unlock 来保护临界区:

pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;void *thread_func(void *arg) {pthread_mutex_lock(&mutex);// 临界区代码pthread_mutex_unlock(&mutex);return NULL;
}

线程终止与回收

当一个线程结束时,其资源不会立即释放。如果需要主线程等待某个子线程结束并回收其资源,可以使用 pthread_join

int pthread_join(pthread_t thread, void **retval);
  • thread:要等待的线程。
  • retval:线程的返回值。

pthread_join 会阻塞当前线程,直到指定的线程结束。

示例代码

以下是一个简单的示例,展示如何在 Linux 下创建多个线程,分别打印不同的消息:

#include <pthread.h>
#include <stdio.h>
#include <unistd.h>void *thread1_func(void *arg) {printf("Thread 1: PID = %d\n", getpid());return NULL;
}void *thread2_func(void *arg) {printf("Thread 2: PID = %d\n", getpid());return NULL;
}int main() {pthread_t thread1, thread2;pthread_create(&thread1, NULL, thread1_func, NULL);pthread_create(&thread2, NULL, thread2_func, NULL);pthread_join(thread1, NULL);pthread_join(thread2, NULL);return 0;
}

查看全部

编译时,使用 -lpthread 链接线程库:

gcc -o thread_example thread_example.c -lpthread

结论

进程和线程是操作系统中的两种基本单位,它们在资源管理和调度上各有不同。进程是资源分配的单位,而线程是调度的基本单位。在 Linux 中,NPTL 库提供了高效的线程创建和管理方式,线程编程能够显著提高程序的并发性。在实际开发中,我们需要通过合适的同步机制来保证线程间的数据一致性。

希望这篇文章能帮助你更好地理解进程和线程,并掌握在 Linux 下的线程编程技巧。

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

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

相关文章

全面指南:使用JMeter进行性能压测与性能优化(中间件压测、数据库压测、分布式集群压测、调优)

目录 一、性能测试的指标 1、并发量 2、响应时间 3、错误率 4、吞吐量 5、资源使用率 二、压测全流程 三、其他注意点 1、并发和吞吐量的关系 2、并发和线程的关系 四、调优及分布式集群压测&#xff08;待仔细学习&#xff09; 1.线程数量超过单机承载能力时的解决…

springboot整合mybatis-plus【详细版】

目录 一&#xff0c;简介 1. 什么是mybatis-plus2.mybatis-plus特点 二&#xff0c;搭建基本环境 1. 导入基本依赖&#xff1a;2. 编写配置文件3. 创建实体类4. 编写controller层5. 编写service接口6. 编写service层7. 编写mapper层 三&#xff0c;基本知识介绍 1. 基本注解 T…

HTTP 常见状态码技术解析(应用层)

引言 HTTP 状态码是服务器对客户端请求的标准化响应标识&#xff0c;属于应用层协议的核心机制。其采用三位数字编码&#xff0c;首位数字定义状态类别&#xff0c;后两位细化具体场景。 状态码不仅是服务端行为的声明&#xff0c;更是客户端处理响应的关键依据。本文将从协议规…

Unity中的键位KeyCode

目录 主要用途 检测按键事件&#xff1a; 处理键盘输入&#xff1a; 基本键位 常用键&#xff1a; 字母键&#xff1a; 数字键&#xff1a; 功能键&#xff1a; 方向键&#xff1a; 控制键&#xff1a; 鼠标键&#xff1a; 其他特殊键&#xff1a; 代码示例 按下…

高考或者单招考试需要考物理这科目

问题&#xff1a;帮忙搜索一下以上学校哪些高考或者单招考试需要考物理这科目的 回答&#xff1a; 根据目前获取的资料&#xff0c;明确提及高考或单招考试需考物理的学校为湖南工业职业技术学院&#xff0c;在部分专业单招时要求选考物理&#xff1b;其他学校暂未发现明确提…

【设计模式】 代理模式(静态代理、动态代理{JDK动态代理、JDK动态代理与CGLIB动态代理的区别})

代理模式 代理模式是一种结构型设计模式&#xff0c;它提供了一种替代访问的方法&#xff0c;即通过代理对象来间接访问目标对象。代理模式可以在不改变原始类代码的情况下&#xff0c;增加额外的功能&#xff0c;如权限控制、日志记录等。 静态代理 静态代理是指创建的或特…

Redis 限流

Target(ElementType.METHOD) Retention(RetentionPolicy.RUNTIME) public interface AccessLimit {/*** 限制次数*/int count() default 15;/*** 时间窗口&#xff0c;单位为秒*/int seconds() default 60; }Aspect Component public class AccessLimitAspect {private static …

Android Coil3缩略图、默认占位图placeholder、error加载错误显示,Kotlin(1)

Android Coil3缩略图、默认占位图placeholder、error加载错误显示&#xff0c;Kotlin&#xff08;1&#xff09; implementation("io.coil-kt.coil3:coil-core:3.1.0")implementation("io.coil-kt.coil3:coil-network-okhttp:3.1.0") <uses-permission …

DeepSeek 助力 Vue 开发:打造丝滑的 键盘快捷键(Keyboard Shortcuts)

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;今天给大家分享一篇文章&#xff01;并提供具体代码帮助大家深入理解&#xff0c;彻底掌握&#xff01;创作不易&#xff0c;如果能帮助到大家或者给大家一些灵感和启发&#xff0c;欢迎收藏关注哦 &#x1f495; 目录 Deep…

uniapp引入uview组件库(可以引用多个组件)

第一步安装 npm install uview-ui2.0.31 第二步更新uview npm update uview-ui 第三步在main.js中引入uview组件库 第四步在uni.scss中引入import "uview-ui/theme.scss"样式 第五步在文件中使用组件

Jmeter进阶篇(34)如何解决jmeter.save.saveservice.timestamp_format=ms报错?

问题描述 今天使用Jmeter完成压测执行,然后使用命令将jtl文件转换成html报告时,遇到了报错! 大致就是说jmeter里定义了一个jmeter.save.saveservice.timestamp_format=ms的时间格式,但是jtl文件中的时间格式不是标准的这个ms格式,导致无法正常解析。对于这个问题,有如下…

React 低代码项目:网络请求与问卷基础实现

&#x1f35e;吐司问卷&#xff1a;网络请求与问卷基础实现 Date: February 10, 2025 Log 技术要点&#xff1a; HTTP协议XMLHttpRequest、fetch、axiosmock.js、postmanWebpack devServer 代理、craco.js 扩展 webpackRestful API 开发要点&#xff1a; 搭建 mock 服务 …

安装海康威视相机SDK后,catkin_make其他项目时,出现“libusb_set_option”错误的解决方法

硬件&#xff1a;雷神MIX G139H047LD 工控机 系统&#xff1a;ubuntu20.04 之前运行某项目时&#xff0c;处于正常状态。后来由于要使用海康威视工业相机&#xff08;型号&#xff1a;MV-CA013-21UC&#xff09;&#xff0c;便下载了并安装了该相机的SDK&#xff0c;之后运行…

人工智能之自动驾驶技术体系

自动驾驶技术体系 自动驾驶技术是人工智能在交通领域的重要应用&#xff0c;旨在通过计算机视觉、传感器融合、路径规划等技术实现车辆的自主驾驶。自动驾驶不仅能够提高交通效率&#xff0c;还能减少交通事故和环境污染。本文将深入探讨自动驾驶的技术体系&#xff0c;包括感…

浅谈模组-相机鬼像

一&#xff0e;前言 在成像中&#xff0c;我们常常会遇到肉眼观测的真实世界中&#xff0c;不存在的异常光影出现在画面中&#xff0c;并伴有各种颜色&#xff0c;我们将这个物体称为鬼像。某些鬼像可能会对图像产生美感的体验&#xff0c;但是大多数的鬼像都会对图像的质量以…

vmware虚拟机Ubuntu Desktop系统怎么和我的电脑相互复制文件、内容

1、先安装vmware workstation 17 player&#xff0c;然后再安装Ubuntu Desktop虚拟机&#xff0c;然后再安装vmware tools&#xff0c;具体可以参考如下视频&#xff1a; VMware虚拟机与主机实现文件共享&#xff0c;其实一点也不难_哔哩哔哩_bilibili 2、本人亲自试过了&…

Spring Boot项目中解决跨域问题(四种方式)

目录 一&#xff0c;跨域产生的原因二&#xff0c;什么情况下算跨域三&#xff0c;实际演示四&#xff0c;解决跨域的方法 1&#xff0c;CrossOrigin注解2&#xff0c;添加全局过滤器3&#xff0c;实现WebMvcConfigurer4&#xff0c;Nginx解决跨域5&#xff0c;注意 开发项目…

Oracle JDK、Open JDK zulu下载地址

一、Oracle JDK https://www.oracle.com/java/technologies/downloads/ 刚进去是最新的版本&#xff0c;往下滑可以看到老版本 二、Open JDK的 Azul Zulu https://www.azul.com/downloads/ 直接可以选版本等选项卡

软件测试:1、单元测试

1. 单元测试的基本概念 单元&#xff08;Unit&#xff09;&#xff1a;软件系统的基本组成单位&#xff0c;可以是函数、模块、方法或类。 单元测试&#xff08;Unit Testing&#xff09;&#xff1a;对软件单元进行的测试&#xff0c;验证代码的正确性、规范性、安全性和性能…

Leetcode.264 丑数 II

题目链接 Leetcode.264 丑数 II mid 题目描述 给你一个整数 n n n &#xff0c;请你找出并返回第 n n n 个 丑数 。 丑数 就是质因子只包含 2 2 2、 3 3 3 和 5 5 5 的正整数。 示例1&#xff1a; 输入&#xff1a;n 10 输出&#xff1a;12 解释&#xff1a;[1, 2, 3,…