news 2026/6/15 17:14:21

Linux:线程的概念、与进程区别及内核实现(线程一)

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Linux:线程的概念、与进程区别及内核实现(线程一)

一、什么是线程?

线程(Thread)是进程内的独立执行流,是操作系统调度的基本单位。简单理解:进程是 “资源分配的容器”,而线程是 “容器里真正干活的执行者”—— 一个进程至少包含一个主线程,多个线程共享进程的全部资源(虚拟地址空间、文件描述符、信号表等),但拥有独立的执行上下文(CPU 寄存器、栈空间)

线程的核心特点:

  1. 轻量级:创建和切换成本远低于进程(无需分配独立资源,仅需初始化执行上下文)
  2. 资源共享:同一进程内的线程共享进程的代码段、数据段、堆、文件描述符等,通信无需跨进程
  3. 独立调度:操作系统以线程为单位分配 CPU 时间片,多个线程可并发执行
  4. 协作性:线程间需通过同步机制(如互斥锁)避免共享资源竞争

二、线程与进程的核心区别(对比表)

对比维度进程(Process)线程(Thread)
资源分配单位独立分配虚拟地址空间、文件描述符等共享所属进程的所有资源
调度执行单位不直接调度,内核调度其主线程内核直接调度的基本单位
上下文切换成本高(需切换页表、地址空间等)低(仅切换 CPU 寄存器、栈)
独立性高(一个进程崩溃不影响其他进程)低(一个线程崩溃可能导致整个进程崩溃)
通信方式复杂(管道、消息队列、共享内存等)简单(共享全局变量、堆内存等)
内核实现(Linux)独立 task_struct + 独立 mm_struct独立 task_struct + 共享 mm_struct

三、Linux 中线程的内核实现:轻量级进程(LWP)

Linux 内核没有专门的线程结构体,线程的实现依赖 “轻量级进程(Lightweight Process, LWP)”—— 无论是进程还是线程,内核都用task_struct(进程控制块)描述,二者的唯一区别是是否共享mm_struct(虚拟地址空间)

  • 进程:一个task_struct对应一个独立的mm_struct(虚拟地址空间),即 “独占资源的主线程”
  • 线程:多个task_struct共享同一个mm_struct,即 “共享资源的执行流”

关键概念补充

  1. 线程组 ID(tgid):同一进程内的所有线程(包括主线程)共享同一个tgid(等于主线程的 PID),ps命令显示的 PID 本质是tgid
  2. 线程 ID(tid):每个线程的task_struct有独立的tid(内核层面的唯一标识),可通过pthread_self()gettid()获取
  3. 内核视角:Linux 内核不区分 “进程” 和 “线程”,只认task_struct,线程的 “共享资源” 特性是通过mm_struct的共享实现的

四、线程的优缺点

优点:

  1. 并发效率高:创建、切换速度快,适合需要大量并发执行的场景(如 Web 服务器处理多请求)
  2. 通信成本低:共享进程资源,无需跨进程通信的复杂机制
  3. 资源利用率高:多个线程可充分利用 CPU 多核资源,提升程序吞吐量

缺点:

  1. 缺乏独立性:线程共享进程资源,一个线程的非法操作(如野指针)可能导致整个进程崩溃
  2. 同步复杂:多个线程访问共享资源时需手动同步(如加锁),否则会出现数据错乱(竞态条件)
  3. 调试难度大:多线程的执行顺序不确定,排查死锁、竞态条件等问题比单线程复杂

五、第一个 Linux 线程程序:Hello World(含代码)

Linux 中线程编程依赖POSIX 线程库(pthread 库),需注意编译时链接-lpthread参数(否则会报未定义引用错误)

代码实现:

#include <stdio.h> #include <pthread.h> #include <unistd.h> // 线程执行函数:线程启动后会执行该函数 void* thread_func(void* arg) { // arg为传递给线程的参数(需强制类型转换) char* thread_name = (char*)arg; printf("Hello World! 我是%s,线程ID:%lu\n", thread_name, (unsigned long)pthread_self()); sleep(1); // 模拟线程执行任务 return NULL; // 线程退出,返回值为NULL } int main() { pthread_t tid; // 存储线程ID int ret; // 1. 创建线程 // 参数:线程ID指针、线程属性(NULL为默认)、线程执行函数、传递给函数的参数 ret = pthread_create(&tid, NULL, thread_func, (void*)"线程1"); if (ret != 0) { perror("pthread_create failed"); return -1; } printf("主线程ID:%lu,创建的线程ID:%lu\n", (unsigned long)pthread_self(), (unsigned long)tid); // 2. 回收线程(等待线程结束,避免线程成为僵尸线程) ret = pthread_join(tid, NULL); if (ret != 0) { perror("pthread_join failed"); return -1; } printf("线程执行完毕,主线程退出\n"); return 0; }

关键说明:

  • pthread_self():获取当前线程的 ID(用户态线程 ID,非内核 tid)
  • pthread_create():创建线程,成功返回 0,失败返回错误码(需用perrorstrerror打印错误信息)
  • pthread_join():回收线程,阻塞主线程直到目标线程结束,避免线程资源泄漏

六、总结

  • 线程是进程内的独立执行流,共享进程资源,调度成本低
  • Linux 中线程本质是轻量级进程,通过task_struct描述,共享mm_struct区分于进程
  • 入门线程编程的核心是掌握 pthread 库的基本使用,理解线程的创建与回收流程

下一篇将详细讲解线程的创建、终止、回收的完整用法,以及线程退出状态的处理,敬请关注!

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/15 14:43:36

DeepSeek-R1-Distill-Llama-8B部署教程:Kubernetes集群中Ollama StatefulSet编排

DeepSeek-R1-Distill-Llama-8B部署教程&#xff1a;Kubernetes集群中Ollama StatefulSet编排 1. 模型介绍与部署准备 DeepSeek-R1系列模型代表了当前开源大语言模型的前沿水平。其中DeepSeek-R1-Distill-Llama-8B是基于Llama架构的蒸馏版本&#xff0c;在保持高性能的同时显著…

作者头像 李华
网站建设 2026/6/14 14:39:05

开源语音合成工具eSpeak NG实用指南:从安装到个性化语音定制

开源语音合成工具eSpeak NG实用指南&#xff1a;从安装到个性化语音定制 【免费下载链接】espeak-ng espeak-ng: 是一个文本到语音的合成器&#xff0c;支持多种语言和口音&#xff0c;适用于Linux、Windows、Android等操作系统。 项目地址: https://gitcode.com/GitHub_Tren…

作者头像 李华
网站建设 2026/6/15 13:17:06

告别复杂代码:Easy-Scraper让数据采集像搭积木一样简单

告别复杂代码&#xff1a;Easy-Scraper让数据采集像搭积木一样简单 【免费下载链接】easy-scraper Easy scraping library 项目地址: https://gitcode.com/gh_mirrors/ea/easy-scraper 你是否曾经面对这样的困境&#xff1a;想要从网页上获取一些数据&#xff0c;却被复…

作者头像 李华
网站建设 2026/6/10 18:15:31

如何让Linux AppImage管理更高效?试试这款一站式解决方案

如何让Linux AppImage管理更高效&#xff1f;试试这款一站式解决方案 【免费下载链接】AppImageLauncher Helper application for Linux distributions serving as a kind of "entry point" for running and integrating AppImages 项目地址: https://gitcode.com/…

作者头像 李华