Linux 理解进程

news2024/11/18 9:24:54

目录

一、基本概念

二、描述进程-PCB

1、task_struct-PCB的一种

2、task_ struct内容分类

三、组织进程

四、查看进程

1、ps指令

2、top命令 

3、/proc文件系统

4、在/proc文件中查看指定进程

5、进程的工作目录 

五、通过系统调用获取进程标示符

1、getpid()/getppid() 

getpid()

getppid()

2、kill命令

五、通过系统调用创建进程

1、bash

2、fork()

父子进程返回结果不同 

 fork之后有两个不同的执行流

fork创建进程后,为什么会有两个返回值?

3、总结:


一、基本概念

程序的本质是文件,以二进制形式存储在磁盘中。当程序文件加载到内存时,就形成了一个进程。因此,操作系统帮助我们将程序转换为进程,以执行特定任务。简单来说,每次打开一个程序或应用时,都会生成一个对应的进程。

  • 教材概念:程序的一个执行实例,正在执行的程序等
  • 内核观点:担当分配系统资源(CPU时间,内存)的实体。

 进程=程序的代码和数据(磁盘)+操作系统维护的进程控制块(PCB)结构体

进程是计算机中正在运行的程序的实例,它包括了程序的代码和数据(存储在磁盘上),以及操作系统维护的进程控制块(PCB)结构体。

  • 代码指的是程序的指令集合,定义了程序的逻辑和执行流程。这些指令存储在磁盘上的可执行文件中。当进程被加载到内存中运行时,这些指令被复制到内存中的代码段中,并由处理器执行。代码段通常包括程序的可执行指令、函数、子程序等。
  • 数据指的是程序在运行过程中使用的变量、常量、对象等存储数据。数据存储在进程的数据段中,也就是进程的内存空间中的一部分。数据段包括全局变量、静态变量、堆、栈等。
  • 这些代码和数据构成了进程的执行环境,使得程序能够在计算机上正确运行。除了代码和数据外,进程还包括了操作系统维护的进程控制块(PCB)。PCB是用于存储和管理进程的各种信息的数据结构,如进程的状态、优先级、寄存器值、内存分配情况等。PCB存储了进程的标识、状态、优先级、程序计数器(PC)值、寄存器内容、内存分配情况等重要信息,是操作系统用于管理和控制进程的关键数据结构。

8a36074687ed4c248de5f31e9b6a3371.png

二、描述进程-PCB

在Windows任务管理器中,我们可以看到许多正在运行和后台运行的进程。操作系统需要对这些进程进行管理,这是通过一种称为进程控制块(PCB)的数据结构来实现的。PCB包含了进程的属性信息,类似于文件的内容和属性。

b9cf05f6a58c40b3a9416964fb4cf005.png

  • 进程信息被放在一个叫做进程控制块的数据结构中,可以理解为进程属性的集合。
  • 课本上称之为PCB(process control block),Linux操作系统下的PCB是: task_struct

1、task_struct-PCB的一种

在Linux中描述进程的结构体叫做task_struct。
  • task_struct是Linux内核的一种数据结构,它会被装载到RAM(内存)里并且包含着进程的信息

2、task_ struct内容分类

  1. 标示符: 描述本进程的唯一标示符,用来区别其他进程。
  2. 状态: 任务状态,退出代码,退出信号等。
  3. 优先级: 相对于其他进程的优先级。(优先级:先后顺序,权限:能与不能)
  4. 程序计数器: 程序中即将被执行的下一条指令的地址。
  5. 内存指针: 包括程序代码和进程相关数据的指针,还有和其他进程共享的内存块的指针。
  6. 上下文数据: 进程执行时处理器的寄存器中的数据[休学例子,要加图CPU,寄存器]。
  7. I/O状态信息: 包括显示的I/O请求,分配给进程的I/O设备和被进程使用的文件列表。
  8. 记账信息: 可能包括处理器时间总和,使用的时钟数总和,时间限制,记账号等。
  9. 其他信息

三、组织进程

可以在内核源代码里找到它。所有运行在系统里的进程都以task_struct双链表的形式存在内核里。
  • 有些操作系统会把任务队列称为任务数组。但是Linux实现时使用的是队列而不是静态数组,所以称为任务队列.

939b2e6aca144776be636364346c8533.png

四、查看进程

新创建一个myproc.c文件,为其编写一个Makefile文件对其进行编译,然后运行myproc这个死循环的可执行程序方便持续查看进程。 

1、ps指令

ps 指令是 Linux 中用于显示当前正在运行的进程的命令。它提供了有关进程的各种信息,例如进程的 PID(进程标识符)、CPU 和内存使用情况等。

  • 默认情况下,如果你只输入 ps 命令而不加任何选项,它会显示与当前终端会话相关的进程信息。这可能包括 Shell 以及你通过该 Shell 启动的其他进程。

2c95448fb2774ed59aa18489dabb5037.png

ps axj | grep 'myproc'

  • 这个命令是将两个命令结合起来使用,首先执行 ps axj 命令,它会显示所有进程的详细信息。然后使用管道 | 将输出传递给 grep 'myproc' 命令,grep 是用于搜索指定模式的文本的工具。
  • 因此,ps axj | grep 'myproc' 命令的作用是从所有进程的详细信息中过滤出包含字符串 'myproc' 的行。这样就能找到与 'myproc' 相关的进程信息。
  • a 选项表示显示所有用户的进程,而不仅仅是当前用户的进程。
  • x 选项表示显示没有控制终端的进程,即显示所有进程而不仅仅是与当前终端关联的进程。
  • j 选项表示以作业控制的格式显示进程信息,这种格式通常包括更多关于进程的详细信息。

4aff602a14414d1dafd8b6713156538d.png

ps axj | head -1 && ps axj | grep 'myproc'

  • 首先执行 ps axj 命令,它会显示所有进程的详细信息。然后使用管道 | 将输出传递给 head -1 命令,head -1 只会显示输出的第一行,即列标题行,这样可以让我们看到列的含义。
  • 接着,&& 运算符表示如果前一个命令成功执行(即返回状态码为 0),则执行下一个命令。因此,如果第一个命令成功执行,就会继续执行第二个命令。
  • 第二个命令是 ps axj | grep 'myproc',它会从所有进程的详细信息中过滤出包含字符串 'myproc' 的行。这样就能找到与 'myproc' 相关的进程信息。

5486aacb0bcb459d83d17ef1488088b5.png

2、top命令 

top 命令是一个交互式的实用程序,用于动态监视系统的进程活动和资源利用情况。它可以显示各个进程的 CPU 占用、内存占用、以及其他系统资源的使用情况。top 命令提供了一个实时更新的进程列表,用户可以通过键盘命令与其交互,例如排序进程、终止进程等。

下面是一些常用的 top 命令的键盘快捷键:

  • q:退出 top 命令。
  • k:终止进程。输入该命令后,会提示输入需要终止的进程的 PID。
  • Space:更新显示。
  • 1:切换到显示每个 CPU 核心的独立信息。
  • M:按内存使用量排序。
  • P:按 CPU 使用量排序。
  • T:按时间排序。

top 命令的输出通常包括:

  • 系统概述:系统的运行时间、平均负载等。
  • 进程信息:各个进程的 PID、CPU 使用率、内存使用量等。
  • 系统资源:CPU 使用率、内存使用情况、交换分区使用情况等。

7bb24e7764f843d2afd71a3b47e95c2b.png

3、/proc文件系统

进程的信息可以通过 /proc 系统文件夹查看,/proc 是一个特殊的虚拟文件系统,用于在Linux系统中提供进程和系统信息。虽然它在文件系统中显示为一个目录,但实际上它是一个由内核动态生成的虚拟文件系统,用于向用户空间提供系统和进程的信息。通过访问 /proc 目录,可以查看和管理运行中的进程,以及获取有关系统状态和配置的信息。

ls /proc

  • 这个目录是动态的,多一个进程就多一个目录,少一个进程就少一个目录。

3586408edd134b86b7f9f67575160873.png

ls /proc -l   

717d5b41f72b4d599e70beb77a665338.png

4、在/proc文件中查看指定进程

查找 myproc 进程: 

ps axj | grep myproc

 使用 ps axj 命令列出所有进程的详细信息,然后用 grep 过滤出包含 myproc 的行。输出显示 myproc 进程的 PID 是 24857,同时也显示了运行 grep 命令本身的进程信息。

在 /proc 目录中查找进程: 

ls /proc -l | grep '24857'

 这个命令列出了 /proc 目录的内容,并使用 grep 查找与 myproc 进程 PID(24857)相关的目录。/proc 文件系统包含了系统信息和正在运行的每个进程的详细信息。每个进程都有一个以其 PID 命名的目录。输出显示了 PID 为 24857 的进程的目录。

列出 /proc/24857 目录的内容: 

ls /proc/24857

acc65acdeef34659a281a5fa31a16b42.png  通过列出 /proc/24857 目录的内容,可以看到与进程相关的许多文件和目录。这些文件和目录提供了关于进程的各种信息,包括:

  • cmdline:进程启动命令。cwd:进程的当前工作目录。environ:进程的环境变量。
  • exe:到启动进程的可执行文件的符号链接。
  • fd:包含指向进程打开的文件描述符的符号链接。
  • status:进程的状态信息。mem:进程的内存映射。
  • 等等其他许多提供关于进程运行状态、资源使用情况等信息的文件和目录。

5、进程的工作目录 

其中cwd表示当前进程的工作目录

9d723de7b25f485abac7b0bcfa85d45d.png

以前学习的FILE *fp = fopen("log.txt","w"),第一个参数没有带路径,默认打开当前路径。为什么呢?

这是因为我们的程序经过编译形成可执行程序,打开文件运行后,形成一个进程,每个进程都会有一个属性,来保存自己的工作路径。

五、通过系统调用获取进程标示符

1、getpid()/getppid() 

 使用man手册查看使用方法 

getpid()

  • 功能getpid() 函数用于获取当前进程的进程 ID(PID)。
  • 返回值:它返回一个 pid_t 类型的值,即当前进程的 PID。在 Linux 中,每个运行的进程都有一个唯一的进程标识符,用于区分不同的进程。
  • 使用场景:该函数在需要根据进程 ID 进行进程管理或信息记录时非常有用。例如,日志记录时可能需要记录发起日志条目的进程 ID。

getppid()

  • 功能getppid() 函数用于获取当前进程的父进程的进程 ID(PPID)。
  • 返回值:同样返回一个 pid_t 类型的值,即当前进程的父进程的 PID。当一个进程被创建(通常是通过 fork() 调用)时,创建它的进程成为其父进程。
  • 使用场景:这在需要了解或控制进程的层级结构时非常有用,比如,在创建守护进程时,可能会根据父进程的 ID 进行特定的逻辑判断。

下面我们创建一个myproc.c文件,输出的pid与我们查询的进程id进行比较: 

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
    while(1)
    {
        pid_t id=getpid(); //获取的是自己的进程
        printf("hello world,pid: &d\n",,id);
        sleep(1);
    }
    return 0;
}

右边复制一个SSH渠道,边运行边查看,如下图:

我们可以发现右边通过getpid成功输出进程myproc的pid。 

我们再输出一下父进程id(ppid) 

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
    while(1)
    {
        pid_t id=getpid();
        printf("hello world, pid: %d, ppid: %d\n",id,getppid());
        sleep(1);
    }
    return 0;
}

输出结果 :

2、kill命令

 kill 命令用于向进程发送信号,以控制进程的行为或终止进程。它的基本语法是: 

kill [options] <PID>

其中 <PID> 是要终止的进程的进程标识符。

常用的 kill 命令选项包括:

  • -l:列出可用的信号名称。
  • -9:发送强制终止信号,也称为 SIGKILL,立即终止目标进程。
  • -15:发送终止信号,也称为 SIGTERM,默认行为是终止目标进程,但它可以被进程捕获、忽略或处理。
  • -SIG<signal>:使用信号名称来指定要发送的信号,例如 -SIGTERM

例如,要终止PID为22053的进程,操作如下:

五、通过系统调用创建进程

1、bash

每次查询当前运行的命令时,我们知道子进程是我们创建并运行的,那父进程是谁?

  • 父进程是bash。

系统中,每个命令行上运行的命令都是由 Bash shell 的一个子进程执行。此外,Bash shell 本身也是系统中许多进程的父进程。 Bash shell 是命令行界面的核心,负责解释和执行用户输入的命令。

复习Shell概念 

  • Shell(外壳)是计算机操作系统中的一个用户界面,用于与操作系统内核进行交互。Shell允许用户通过命令行或图形用户界面(GUI)来执行操作系统提供的功能和应用程序。Shell接受用户输入的命令,并将其解释并传递给操作系统内核执行。
  • 在类Unix操作系统(如Linux、macOS等)中,常见的Shell包括Bash(Bourne Again Shell)、Zsh(Z Shell)、Fish等。这些Shell提供了丰富的命令和功能,使用户能够管理文件、运行程序、配置系统等操作。

在大多数Linux系统中,常用的命令行解释器(shell)是Bash(Bourne Again Shell)。Bash是一个流行的Unix shell,也是许多Linux发行版的默认shell。它提供了丰富的功能和命令,使用户能够与操作系统进行交互、执行命令、编写脚本等。

如果我们kill -9 bash呢?

kill -9 加上bash的id ,命令行会会失效。

  1. 立即终止会话: 当前终端或会话中的Bash进程会被立即终止,导致您失去对该终端会话的访问。

  2. 子进程可能被终止: 如果该Bash进程启动了任何子进程,那么这些进程可能也会因为父进程的终止而受到影响,除非它们已经被设置为守护进程或与终端会话分离。

  3. 未保存的数据丢失: 如果您在该Bash会话中运行的程序或编辑器中有未保存的工作,强制终止Bash进程将导致这些数据丢失。

  4. 不影响其他Bash会话: 如果您在同一系统上打开了多个终端会话,这个操作只会影响您向其发送kill -9命令的那个Bash进程。其他Bash会话及其子进程将不受影响。

  5. 可能需要重新登录: 终止当前会话的Bash进程后,您可能需要重新打开一个新的终端会话或重新登录您的用户账户来恢复正常操作。

  • 注意:每次登入的bash都不同,复制一个SSH通道,杀掉bash进程,不会影响另一个窗口运行命令。

2、fork()

在Linux系统中,fork() 是一个系统调用,用于创建一个新的进程。调用 fork() 后,操作系统会复制当前进程的副本,包括代码段、数据段、堆栈等,然后在新的进程中运行。原始进程被称为父进程,新创建的进程被称为子进程。

  • 在 fork() 调用后,父进程和子进程会继续执行后续的代码,但是它们会有不同的进程ID(PID)和返回值。在父进程中,fork() 返回子进程的PID,而在子进程中,fork() 返回0。
  • 通过 fork() 创建子进程后,父子进程之间会共享一些资源,如文件描述符、内存映射等,但各自有独立的地址空间。这样可以实现并发执行,父子进程可以在独立的执行环境中运行不同的任务。 

按照顺序先运行没有fork的代码,再运行有fork的版本。 

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>

int main()
{
    printf("I am parent process.\n");

    printf("you can see me.\n");

    return 0;
}
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>

int main()
{
    printf("I am parent process.\n");
    fork();
    printf("you can see me.\n");
    sleep(1);

    return 0;
}

 输出结果如下:我们可以发现创建子进程后有一句被输出了两次。

 

这是因为创建进程后,变为两个程序,一个是父进程,另一个是子进程,他们分别运行fork()后面的程序。

父子进程返回结果不同 

我们再观察下面代码的输出结果 

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>

int main()
{
    printf("I am parent process.\n");
    pid_t ret=fork();
    
    printf("ret: %d, pid: %d, ppid: %d\n",ret,getpid(),getppid());
    sleep(1);

    return 0;
}

fork为什么给父进程返回子进程pid,给子进程返回0?

  • fork()在父进程中被调用时,它会返回子进程的PID(进程ID),这样父进程就可以知道新创建的子进程的PID,从而可以对子进程进行管理或跟踪。
  • fork()在子进程中被调用时,它会返回0,这样子进程可以通过返回值来确定自己是子进程,而不是父进程。子进程可以根据返回值执行自己的逻辑,而不是继续执行父进程的代码。

 fork之后有两个不同的执行流

为什么下面else if和else可以同时输出呢?因为fork之后有两个不同的执行流 。

#include <stdio.h>
#include <unistd.h>

int main()
{
    pid_t id=fork();
    if(id<0)
    {
        //创建失败
        perror("fork");
        return 1;
    }
    else if(id==0)
    {
        //child process
        while(1)
        {
            printf("I am child, pid:%d, ppid:%d\n",getpid(),getppid());
            sleep(1);
        }
    }
    else
    {
        //parent process
        while(1)
        {
            printf("I am father, pid:%d, ppid:%d\n",getpid(),getppid());
            sleep(1);
        }   
    }
    printf("you can see me\n");
    sleep(1);//进程退出顺序不一样
    return 0;
}

 我们通过下面命令是用来监视名为"myproc"的进程。

while :; do ps axj | head -1 && ps axj | grep myproc | grep -v grep; sleep 1; done

这是一个无限循环(while :; do ... done),在循环内部,它首先使用 ps axj | head -1 命令来显示进程列表的标题行(列名),然后通过 ps axj | grep myproc | grep -v grep 命令来显示包含 "myproc" 的进程,同时使用 grep -v grep 来排除 grep 进程本身。最后,sleep 1 用于每秒暂停一次

我们可以为每次输出结果之间加上一条分割线。(###分隔符过长仅保留部分)

while :; do ps axj | head -1 && ps axj | grep myproc | grep -v grep; sleep 1; echo "###"; done

fork创建进程后,为什么会有两个返回值?

  • Linux中的fork()系统调用返回两次值,这是因为在Unix系统中,fork()的设计是基于父子进程的概念。调用fork()时,操作系统会复制当前进程,创建一个新的子进程。这个子进程是父进程的副本,但它们是两个独立的进程,各自有自己的地址空间。
  • fork()创建进程后返回两次值的原因是,在fork()内部,父进程和子进程各自会执行自己的return语句,这导致了两次返回。这并不意味着操作系统保存了两次状态,而是因为父进程和子进程在不同的执行上下文中各自返回。

3、总结:

  • 在操作系统中,当需要运行一个进程时,实际上是从任务结构(task_struct)形成的队列中选择一个任务结构来执行该进程的代码。进程调度实质上就是在任务结构队列中选择一个进程的过程。
  • 在创建父子进程后,哪个进程先运行并不确定。这是由操作系统的调度器决定的。调度器会根据一定的策略(如优先级、时间片轮转等)来决定下一个要运行的进程。
  • 因此,父子进程的执行顺序不确定,取决于操作系统的调度算法。

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

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

相关文章

空间复杂度(数据结构)

概念&#xff1a; 空间复杂度也是一个数学表达式&#xff0c;是对一个算法在运行过程中临时占用存储空间大小的量度 。 空间复杂度不是程序占用了多少bytes的空间&#xff0c;因为这个也没太大意义&#xff0c;所以空间复杂度算的是变量的个数。空间复杂度计算规则基本跟实践复…

nicegui学习使用

https://www.douyin.com/shipin/7283814177230178363 python轻量级高自由度web框架 - NiceGUI (6) - 知乎 python做界面&#xff0c;为什么我会强烈推荐nicegui 秒杀官方实现&#xff0c;python界面库&#xff0c;去掉90%事件代码的nicegui python web GUI框架-NiceGUI 教程…

EI级 | Matlab实现PCA-GCN主成分降维结合图卷积神经网络的数据多特征分类预测

EI级 | Matlab实现PCA-GCN主成分降维结合图卷积神经网络的数据多特征分类预测 目录 EI级 | Matlab实现PCA-GCN主成分降维结合图卷积神经网络的数据多特征分类预测效果一览基本介绍程序设计参考资料 效果一览 基本介绍 1.Matlab实现PCA-GCN主成分降维结合图卷积神经网络的数据多…

用conda创建虚拟环境

下载好conda之后&#xff0c;在跑代码之前&#xff0c;可以用conda来创建虚拟环境&#xff0c;然后在虚拟环境中下载包pip之类的。 创建步骤如下&#xff1a; 1.conda create --name hhh 其中hhh为我的虚拟环境的名字&#xff0c;之后选择y即yes即可继续创建 可以看到&#…

LVS集群 ----------------(直接路由 )DR模式部署 (二)

一、LVS集群的三种工作模式 lvs-nat&#xff1a;修改请求报文的目标IP,多目标IP的DNAT lvs-dr&#xff1a;操纵封装新的MAC地址&#xff08;直接路由&#xff09; lvs-tun&#xff1a;隧道模式 lvs-dr 是 LVS集群的 默认工作模式 NAT通过网络地址转换实现的虚拟服务器&…

springcloud第3季 consul服务发现注册,配置中心2

一 consul的作用 1.1 为何使用注册中心 为何要用注册中心&#xff1f; 1.A服务调用B服务&#xff0c;使用ip和端口&#xff0c;如果B服务的ip或者端口发生变化&#xff0c;服务A需要进行改动&#xff1b; 2.如果在分布式集群中&#xff0c;部署多个服务B&#xff0c;多个服…

【开发工具】认识Git | 认识工作区、暂存区、版本库

文章目录 一、Git初识git本质上是一个版本控制器 二、Git的安装 - CentOS三、Git基本操作1. 创建Git本地仓库2. 配置Git3. 认识工作区、暂存区、版本库4. 版本回退5. 撤销修改情况1&#xff1a;对于工作区的代码&#xff0c;还没有add情况二&#xff1a;已经add &#xff0c;但…

有哪些平台可以赚些零花钱?分享7个副业兼职平台

正规可靠的兼职副业平台有很多&#xff0c;以下是一些常见的平台&#xff1a; 1&#xff0c;微头条 微头条是一种短文本分享平台&#xff0c;通过精简和优化文字&#xff0c;以吸引读者的注意力。需要在有限的字数内表达清晰明了的观点&#xff0c;关键词的准确使用是关键。例…

不允许你不知道Python作用域

在Python中&#xff0c;变量的作用域限制非常重要。根据作用域分类&#xff0c;有局部、全局、函数和内建作用域。无作用域限制的变量可以在分支语句和循环中定义&#xff0c;并在外部直接访问。不同的作用域决定了变量的可访问范围&#xff0c;访问权限取决于变量的位置。 1.…

面试经典150题 -- 图的广度优先遍历 (总结)

总的链接 面试经典 150 题 - 学习计划 - 力扣&#xff08;LeetCode&#xff09;全球极客挚爱的技术成长平台 909 . 蛇梯棋 链接 : . - 力扣&#xff08;LeetCode&#xff09; 题意 &#xff1a; 直接bfs就好了 &#xff0c; 题意难以理解 : class Solution:def snakesA…

虚拟机中安装Win98

文章目录 一、下载Win98二、制作可启动光盘三、VMware中安装Win98四、Qemu中安装Win981. Qemu的安装2. 安装Win98 Win98是微软于1998年发布的16位与32位混合的操作系统&#xff0c;也是一代经典的操作系统&#xff0c;期间出现了不少经典的软件与游戏&#xff0c;还是值得怀念的…

office办公软件太贵了 Microsoft的Word为什么要买 Microsoft365家庭版多少钱 Microsoft365密钥

Microsoft office是一个被广泛使用的办公软件&#xff0c;它包括了 Word、Excel、PowerPoint 等多种常用的应用程序&#xff0c;已成为许多企业、机构和个人必备的工具。 首先&#xff0c;要理解 Microsoft Office 的价格&#xff0c;我们需要考虑到它的功能和市场需求。Micro…

Pycharm使用教程

1.设置字体型号与大小 file->setting->editor->font(字型)&#xff0c;size&#xff08;大小&#xff09; 2.设置背景颜色 file->setting->editor->color scheme->scheme 3.注释/取消注释 ctrl/ 选中需要注释的部分&#xff0c;双击ctrl/ 取消注释则选…

揭秘数据中心幕后:从电力消耗到温度调控的策略

建设并运营数据中心并非简单的连接硬盘、通电和联网就可以&#xff0c;而是涉及复杂的硬件集成、能源管理、散热设计以及适应不断增长的数据处理和存储需求等诸多挑战。随着全球互联网的普及和AI技术的快速发展&#xff0c;数据中心的规模和能耗需求都在急剧增加。尤其是在电力…

Vue.js计算属性:实现数据驱动的利器

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…

新雀优化算法NOA求解机器人栅格地图最短路径规划,可以自定义地图(提供MATLAB代码)

一、星雀优化算法 星雀优化算法(Nutcracker optimizer algorithm,NOA)由Mohamed Abdel-Basset等人于2023年提出&#xff0c;该算法模拟星雀的两种行为&#xff0c;即&#xff1a;在夏秋季节收集并储存食物&#xff0c;在春冬季节搜索食物的存储位置。CEC2005:星雀优化算法(Nut…

判断链表回文

题目&#xff1a; //方法一&#xff0c;空间复杂度O(n) class Solution { public:bool isPalindrome(ListNode* head) {vector<int> nums; //放进数组后用双指针判断ListNode* cur head;while(cur){nums.emplace_back(cur->val);cur cur->next;}for(int i0…

Spring MVC 全局异常处理器

如果不加以异常处理&#xff0c;错误信息肯定会抛在浏览器页面上&#xff0c;这样很不友好&#xff0c;所以必须进行异常处理。 1.异常处理思路 系统的dao、service、controller出现都通过throws Exception向上抛出&#xff0c;最后由springmvc前端控制器交由异常处理器进行异…

【玩转Linux】有关Linux权限

目录 一.Linux权限的概念 1. 权限的本质 2.Linux中的用户 3.Linux中的权限管理 (1)文件访问者的分类 (2)文件类型和访问权限&#xff08;事物属性&#xff09; ①文件基本权限 ②文件权限值的表示方法 (3)文件访问权限的相关设置方法 ① 用 户 表 示 符 / - 权 …

BIM技术趋势

随着 2024 年的到来&#xff0c;3D 建模和人工智能 (AI) 的集成将重新定义建筑信息模型 (BIM) 的格局。 人工智能驱动的解决方案将为设施管理、照明设计、工程和建筑等 BIM 相关领域树立新标准。 本文重点介绍 3D 建模和人工智能的这些趋势如何比更传统的方法提供竞争优势。 它…