由浅入深的了解进程(3)

进程状态

每一个进程都拥有自己的代码和数据,同时也具备自己的状态。

1、Linux中的进程状态

Linux系统中,进程状态是通过struct task_struct结构体中的各种属性和变量来表示的。在Linux内核的源代码中,进程状态的定义如下:

/* * The task state array is a strange "bitmap" of * reasons to sleep. Thus "running" is zero, and * you can test for combinations of others with * simple bit tests. */ static const char * const task_state_array[] = { "R (running)", /* 0 */ "S (sleeping)", /* 1 */ "D (disk sleep)", /* 2 */ "T (stopped)", /* 4 */ "t (tracing stop)", /* 8 */ "X (dead)", /* 16 */ "Z (zombie)", /* 32 */ };

其中,task_struct结构体用数组形式表示进程的各种状态。

1、1、进程状态R+和S+的执行代码源码

以下是一个示例代码:

#include #include #include int main() {   while(1)   {     printf("I am a process,pid:%dn",getpid());                                                  }  return 0; }

由浅入深的了解进程(3)

当我们运行一个可执行程序时,可能会发现进程状态显示为S。这与我们理解的S状态(休眠)似乎有所不同。让我们尝试另一个代码:

#include #include #include int main() {   while(1)   {     //printf("I am a process,pid:%dn",getpid());                                                  }  return 0; }

由浅入深的了解进程(3)

奇怪的是,为什么这次运行显示的状态是R呢?首先,R状态表示进程正在运行,而S状态表示进程处于休眠状态。添加printf语句后,为什么进程状态变成了S呢?这是因为printf会将信息打印到屏幕上。根据冯诺依曼体系结构,屏幕作为外设是硬件,CPU处理完信息后需要先存储在内存中。由于CPU的速度远快于屏幕,进程在等待屏幕输出结束时会进入S状态。因此,我们在查看状态时,大多数情况下会看到S状态而不是R状态。如果不需要通过外设,直接查询的状态通常会是R状态。这也说明了外设与CPU速度的巨大差距。因此,S+状态更准确的描述应该是:进程在等待“资源”就绪。此外,这种状态也能够随时中断睡眠。

实际上,如果我们使用以下命令:

./testStatus &:如果这样操作的话,进程就会在后端运行,此时S状态而不是S+状态了

其中的“+”表示在前台运行,如果有此符号,则说明进程在前台运行;如果没有,则说明在后台运行。

1、2、进程状态T/t

由浅入深的了解进程(3)

为了观察进程是否能进入T状态,我们可以使用kill命令。kill中的19号信号是signal stop,表示暂停进程。

由浅入深的了解进程(3)

此时,进程被暂停,等待进一步唤醒。18号信号SIGCONT,表示信号继续。发送此信号后,进程将恢复运行。

由浅入深的了解进程(3)由浅入深的了解进程(3)

kill -9/-19/-18分别表示杀掉进程、暂停进程和继续进程。进程暂停在实际开发中很常见,尤其是在调试代码时使用断点。

由浅入深的了解进程(3)

在未开始运行时,只存在gdb的S状态。当我们设置断点并运行时,gdb的状态从S变为R,而程序本身的状态也从无变为t。这是因为程序遇到断点时需要暂停。

1、3、进程状态D

Linux系统特有的状态。源码中,D代表disk sleep。如果进程中的内存严重不足,Linux操作系统有权杀掉进程来释放空间。如果需要将一个非常重要的1GB资源存储到硬盘上,由于硬件速度远低于CPU处理速度,内存中的1GB数据可能大多数时间处于S状态。如果此时内存严重不足,操作系统可能会直接杀掉进程以释放空间,导致硬盘处理完后找不到数据,重要数据丢失。为了防止这种情况,我们需要给进程设置一个状态,凡是要向磁盘写入数据的进程,即使是操作系统也不能删除,而应删除其他进程。因此,进行数据I/O的进程需要加上D状态,以避免被操作系统删除。D状态表示深度睡眠,不可被暂停。进程如何取消?1、重启(断电);2、自己醒来。

1、4、进程状态X和Z

X状态是死亡状态,比较难以查看且瞬时。类似于现实生活中的查案,警察来到现场不会立即收拾现场,而是保持现场,等待所有流程走完,取证完毕后才收拾。同样,进程结束后不会立即退出,而是处于一种僵尸状态Z,如果父进程不对其进行回收,进程将一直处于Z状态,不会进入X状态。

知了zKnown 知了zKnown

知了zKnown:致力于信息降噪 / 阅读提效的个人知识助手。

知了zKnown 65 查看详情 知了zKnown

2、僵尸进程

由浅入深的了解进程(3)

此代码的含义是父进程一直处于while循环中,而子进程只执行5次。子进程执行结束后,父进程尚未结束,因此子进程处于未被回收的状态,即僵尸进程。

由浅入深的了解进程(3)

其中defunct表示失效的,不存在的。这意味着子进程确实已经退出,没有数据和代码,但仍存在PCB结构体。僵尸进程表示:已经运行完毕,但需要维持自己的退出信息,记录在task_struct中,等待父进程读取。如果没有父进程读取,僵尸进程将一直存在。

3、孤儿进程

顾名思义,孤儿进程是指父进程先退出,而子进程尚未结束。最典型的表现是:

由浅入深的了解进程(3)

如果父进程先退出,子进程将变成孤儿进程,孤儿进程将被1号进程(OS)领养,以确保孤儿进程能够被正确处理。

4、bash概括

我们以前启动的所有进程,为什么从来没有关心过僵尸进程和内存泄漏?因为在命令行中启动的子进程,其父进程是bash,bash会自动回收新进程的Z状态。

5、进程的阻塞、挂起和运行

5、1、运行

由浅入深的了解进程(3)

当多个task_struct排成队列放在CPU中时,该进程即为R状态(运行状态)。一个进程一旦持有一个CPU,会一直运行到进程结束吗?不会。因为是基于时间片进行调用,如果一个进程超过规定的时间片,就需要跳过该进程,先执行下一个,以此类推。这种基于时间片的调度方式称为并发,防止CPU被死循环占用。只有在多个CPU的情况下,才能实现真正的并行算法调度。但Linux的调度算法不止这一种。

5、2、阻塞问题

1、在C语言中使用scanf时,如果当前没有对键盘进行输入,此时的进程处于什么状态?scanf在等待什么资源?此时scanf处于阻塞状态,在Linux中相当于S+状态,等待键盘资源是否就绪(键盘是否输入数据)。

2、操作系统如何对硬件进行管理?

由浅入深的了解进程(3)

操作系统对硬件的管理不是直接管理硬件,而是管理其数据。每个硬件都有自己的结构体供操作系统控制。硬件的结构体能够有程序的等待队列。当进程队列中有scanf时,需要检查硬件是否准备好,如果没有准备好,就需要将该进程剥离出来,排到硬件的队列中,等待硬件数据输入成功。因此,不仅CPU有自己的运行队列,各种硬件也有自己的运行队列。

由浅入深的了解进程(3)

这种进程状态被称为阻塞。当硬件操作完成后,操作系统会将硬件结构体中的队列中的特定进程释放,重新链回相应的进程运行队列中。

5、3、挂起

磁盘中存在swap分区。当操作系统内存紧张时,需要将一些不在运行队列中的进程的代码和数据暂时移出到磁盘的swap分区中。因为此时的进程还在硬件的排序队列中,数据和代码暂时不会被访问。这样可以优化内存容量。当读取数据结束,重新回到运行队列时,再从swap分区中唤出相应的代码和数据。此时进程仍然存在,只是数据和代码放在磁盘中。这种情况称为阻塞挂起态。但频繁的唤出和唤入会降低进程的效率。

以上就是由浅入深的了解进程(3)的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/465082.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月8日 05:07:00
下一篇 2025年11月8日 05:08:05

相关推荐

  • 正确使用Python verify-email 库处理邮件验证结果

    Python的verify-email库在进行邮件地址验证时,不会通过抛出异常来指示验证失败,而是通过其核心函数verify_email()返回布尔值(True表示有效,False表示无效)。本文将详细指导如何正确地利用这一机制,通过条件判断来处理邮件验证结果,而非尝试捕获不存在的VerifyEma…

    2025年12月14日
    000
  • 使用Tkinter自定义类实现带滚动条的TreeView

    本文将介绍如何在Tkinter中使用自定义类创建带滚动条的TreeView控件。摘要如下:本文介绍了如何在使用Tkinter自定义类创建TreeView控件时正确集成滚动条。关键在于确保将父控件传递给ttk.Treeview的初始化函数,并正确配置滚动条与TreeView的关联。同时,通过设置fil…

    2025年12月14日
    000
  • Tkinter自定义Treeview与滚动条集成:解决布局错位问题

    本文旨在解决Tkinter中自定义Treeview组件与滚动条集成时常见的布局错位问题。核心原因在于自定义类在初始化时未正确传递父组件,导致组件层级混乱。教程将详细阐述如何通过在super().__init__()中传递父组件,并结合pack()布局管理器中的fill和expand参数,确保滚动条与…

    2025年12月14日
    000
  • Tkinter 自定义 Treeview 类与滚动条集成指南

    本文档旨在指导开发者如何在 Tkinter 中创建一个自定义的 Treeview 类,并正确地集成垂直和水平滚动条。通过修改 super().__init__() 的调用方式,将父窗口传递给父类的构造函数,以及调整 pack() 方法的参数,可以解决滚动条位置不正确的问题,并实现 Treeview …

    2025年12月14日
    000
  • Tkinter自定义Treeview与滚动条的正确集成指南

    本文探讨了在Tkinter中使用自定义ttk.Treeview类时,滚动条位置异常的问题。核心原因是未将父组件正确传递给基类的构造函数,导致组件层级错误。通过在super().__init__()中传入父组件,并优化pack()布局参数,可以有效解决此问题,确保滚动条与Treeview的正确关联和显…

    2025年12月14日
    000
  • Tkinter自定义Treeview与滚动条的正确集成方法

    本文将探讨在Tkinter中创建自定义Treeview类时,滚动条位置异常的常见问题及其解决方案。核心在于确保自定义组件在初始化时正确指定其父容器,避免其默认成为根窗口的子组件。通过修正构造函数中的父容器传递,并结合合理的布局管理(如pack的fill和expand选项),可以实现滚动条与Treev…

    2025年12月14日
    000
  • 在social-auth-app-django中通过自定义字段实现社交账户关联

    本教程详细介绍了如何在social-auth-app-django中,为具有自定义字段(如telegram_id)的UserModel实现社交账户的智能关联。通过自定义SOCIAL_AUTH_PIPELINE中的函数,我们可以在用户首次通过社交平台(如Telegram)登录时,根据自定义字段检查现有…

    2025年12月14日
    000
  • 使用 Django Social Auth 通过自定义字段关联社交账号

    本文档介绍了如何在 Django 项目中使用 python-social-auth 库,通过自定义字段(例如 Telegram ID)将社交账号与用户模型关联。我们将创建一个自定义的 pipeline,在用户通过 Telegram 登录时,根据 telegram_id 字段查找已存在的用户,并将其与…

    2025年12月14日
    000
  • 在Django Social Auth中通过自定义字段关联用户模型

    本教程详细阐述了如何在Django Social Auth中,通过自定义用户模型字段(如Telegram ID)实现用户关联。当标准关联策略不适用时,通过创建并集成自定义管道函数,可以检查现有用户模型中是否存在匹配的自定义字段,并据此关联社交账户,从而确保用户登录流程的灵活性和准确性,避免重复创建用…

    2025年12月14日
    000
  • 使用 Python 进行网页数据抓取:基础教程

    本文档旨在提供一个关于如何使用 Python 进行网页数据抓取的基础教程。我们将使用 requests 库获取网页内容,并使用 BeautifulSoup 库解析 HTML。本文将提供代码示例,并强调在进行网页抓取时需要注意的道德和法律问题,以及应对网站结构变化的方法。 网页抓取基础 网页抓取(We…

    2025年12月14日
    000
  • 解决Flask CORS问题:从配置到FastAPI的迁移

    本文旨在帮助开发者解决在使用Flask框架时遇到的CORS(跨域资源共享)问题。通过分析常见错误配置和请求头,提供详细的排查步骤。同时,考虑到flask-cors可能存在的问题,本文推荐使用FastAPI框架作为替代方案,并提供快速迁移的示例代码,以确保跨域请求的顺利进行。 CORS(跨域资源共享)…

    2025年12月14日
    000
  • 构建双服务器通信:Laravel 与 Python Flask 的异步请求处理

    本文旨在解决 Laravel 服务器和 Python Flask 服务器之间进行双向通信时,避免阻塞连接的问题。通过探讨传统 HTTP 服务器的局限性,介绍了使用异步编程模型(如 asyncio 和 aiohttp)来优化服务器性能的方法。文章将重点讲解如何在 Flask 框架中利用异步特性,以及如…

    2025年12月14日
    000
  • 使用异步请求在 Laravel 和 Flask 服务器之间进行通信

    本文档介绍了如何在 Laravel (PHP) 和 Flask (Python) 服务器之间实现非阻塞的双向请求通信。传统 HTTP 服务器的线程模型限制了并发处理能力,当一个服务器需要等待另一个服务器的响应时,会阻塞当前线程。本文将探讨使用异步编程解决此问题的方法,重点介绍如何在 Flask 中利…

    2025年12月14日
    000
  • 解决 Flask 中 Axios POST 请求的 CORS 错误

    本文旨在解决在使用 Flask 作为%ignore_a_1%,React 作为前端时,由于 CORS (跨域资源共享) 策略导致的 Axios POST 请求失败的问题。文章将分析常见原因,并提供一种通过切换到 FastAPI 框架来规避此问题的解决方案,同时简要介绍 FastAPI 中 CORS …

    2025年12月14日
    000
  • 使用异步方式在 Laravel 和 Flask 服务器之间进行通信

    本文档介绍了如何在 Laravel 和 Python Flask 服务器之间实现非阻塞的请求通信。针对机器学习任务,Flask 服务器需要从 Laravel 服务器获取最新数据,传统同步方式会阻塞连接。本文将探讨使用异步编程解决此问题,重点介绍 asyncio 和 aiohttp,并提供示例代码和注…

    2025年12月14日
    000
  • 实现服务器间非阻塞通信:Python Flask与Laravel的异步交互策略

    本文探讨了在Python Flask和Laravel服务器之间进行数据交互时,如何避免传统阻塞式请求导致的性能瓶颈。核心解决方案是采用异步I/O模型,特别是利用Python的asyncio和aiohttp库,或支持异步的Web框架(如Flask 3.0+或Starlette),以实现服务器线程在等待…

    2025年12月14日
    000
  • Faiss-GPU 安装失败问题排查及解决方案

    本文旨在解决在 Python 3.8 环境下使用 pip 安装 faiss-gpu 库时遇到的问题。文章将分析安装失败的常见原因,并提供一种基于编译 faiss-cpu 的替代方案,以成功启用 GPU 支持,从而避免直接编译 faiss-gpu 源码可能遇到的错误。 Faiss-GPU 安装问题分析…

    2025年12月14日
    000
  • 使用异步请求在 Laravel 和 Flask 服务器间进行通信

    本文探讨了如何在 Laravel 和 Python Flask 服务器之间实现非阻塞的请求通信。传统的 HTTP 服务器模型在处理请求时会阻塞线程,影响性能。本文介绍了两种解决方案:使用多线程/进程,以及采用异步服务器架构。重点讲解了如何利用 asyncio 和 aiohttp 等库,将 Flask…

    2025年12月14日
    000
  • Faiss-GPU 安装指南:解决 Python 3.8 版本安装失败问题

    本文将解决在 Python 3.8 环境下使用 pip 安装 faiss-gpu 库时遇到的 “inconsistent name: expected ‘faiss-gpu’, but metadata has ‘faiss-cpu’” 错…

    2025年12月14日
    000
  • Faiss-GPU 安装问题解决指南(Python 3.8)

    本文旨在解决在使用 pip 安装 faiss-gpu 时遇到的常见问题,尤其是在 Python 3.8 环境下。通过分析错误信息,明确问题根源在于 faiss-gpu 的元数据不一致。文章将提供一种替代方案,通过设置环境变量并安装 faiss-cpu 来启用 GPU 支持,从而绕过直接编译 fais…

    2025年12月14日
    000

发表回复

登录后才能评论
关注微信