Linux进程基础知识

1. 什么是进程?

  • 进程就是运行中的程序,是系统资源分配的基本单位
  • 每个进程都有唯一的PID(进程ID)
  • 进程有父子关系,通过ps -ef可以查看

2. 进程的创建 - fork()

复制代码
pid_t pid = fork();
  • 简单理解:fork()就像细胞分裂,一个变两个
  • 调用一次,返回两次:父进程得到子进程PID,子进程得到0
  • 子进程是父进程的复制品

3. 进程的程序替换 - exec族

复制代码
execl("/bin/ls", "ls", "-l", NULL);
  • 保留进程ID,但内容完全变成新程序
  • 就像换灵魂:外壳不变,内在全变
  • 常用函数:execl, execlp, execle, execv, execvp等

4. 进程的等待与回收 - wait/waitpid

复制代码
pid_t wait(int *status);

pid_t waitpid(pid_t pid, int *status, int options);
  • 父进程等待子进程结束并回收资源
  • 避免僵尸进程(zombie)的产生
  • waitpid比wait更灵活,可以非阻塞等待

5. 进程的退出

exit(0); // 标准C库函数

_exit(0); // 系统调用

  • exit会刷新缓冲区,exit直接退出
  • 返回值0表示正常退出

一. 文件操作与I/O

1. 文件描述符

  • 就像文件的"身份证号码"
  • 标准输入(0)、标准输出(1)、标准错误(2)
  • 每个进程最多可打开的文件数有限制

2. 文件操作基本函数

复制代码
// 打开文件

int fd = open("test.txt", O_RDWR | O_CREAT, 0644);

// 读取文件

char buf[1024];

int n = read(fd, buf, sizeof(buf));

// 写入文件

write(fd, "hello", 5);

// 关闭文件

close(fd);

// 文件指针定位

lseek(fd, 0, SEEK_SET);  // 回到文件开头

3. 文件属性获取

复制代码
struct stat st;

stat("file.txt", &st);



- 获取文件大小:st.st_size
  • 判断文件类型:S_ISREG(st.st_mode)、S_ISDIR(st.st_mode)

4. 目录操作

复制代码
DIR *dir = opendir(".");

struct dirent *entry;

while ((entry = readdir(dir)) != NULL) {

    printf("%s\n", entry->d_name);

}

closedir(dir);

三、进程间通信(IPC)

1. 管道通信

复制代码
int fd[2];

pipe(fd);  // fd[0]读端,fd[1]写端
  • 单向通信:水管模型,一端进一端出
  • 只能用于有亲缘关系的进程
  • 适合简单的数据传输

2. 命名管道(FIFO)

复制代码
// 创建命名管道

mkfifo("myfifo", 0644);

// 使用方法与普通文件类似

int fd = open("myfifo", O_WRONLY);
  • 可用于无关进程间通信
  • 遵循先进先出原则

3. 信号通信

复制代码
// 信号处理函数

void sig_handler(int signo) {

    printf("收到信号: %d\n", signo);

}

// 设置信号处理

signal(SIGINT, sig_handler);
  • 常见信号:
  • SIGINT (2): Ctrl+C中断
  • SIGKILL (9): 强制终止(不可捕获)
  • SIGTERM (15): 终止信号
  • SIGCHLD (17): 子进程状态改变
  • SIGALRM (14): 定时器到期

四、实用示例

1. 创建守护进程

复制代码
void daemonize() {

    pid_t pid = fork();

    if (pid < 0) exit(1);

    if (pid > 0) exit(0);  // 父进程退出

    

    setsid();  // 创建新会话

    

    // 关闭标准输入输出错误

    close(0);

    close(1);

    close(2);

    // ...更多操作

}

2. 简单shell实现

复制代码
while (1) {

    printf("myshell> ");

    fgets(cmd, sizeof(cmd), stdin);

    

    if (fork() == 0) {

        // 子进程执行命令

        execlp(cmd, cmd, NULL);

        exit(0);

    } else {

        // 父进程等待

        wait(NULL);

    }

}

五、常见错误与调试技巧

1. 经典错误

  • 僵尸进程:子进程结束但未被回收
  • 孤儿进程:父进程先于子进程结束
  • 内存泄漏:申请的内存未释放
  • 文件描述符泄漏:打开文件未关闭

2. 调试工具

  • strace:跟踪系统调用
  • gdb:代码调试器
  • valgrind:内存错误检测
  • lsof:查看打开的文件
相关推荐
爱奇艺技术产品团队1 小时前
服务性能防腐体系:基于自动化压测的熔断机制
运维·自动化
小金三岁半1 小时前
云服务器安装redis
运维·服务器·redis
电商数据girl2 小时前
在1688平台上如何实现铺货和上传商品的自动化?
运维·自动化
已是上好佳2 小时前
Tcp网络通信的基本流程梳理
linux·运维·服务器·c++
带鱼吃猫2 小时前
Linux系统之美:进程初识
linux·服务器
m0_555762902 小时前
qt designer中的Spacer相关设置
服务器·开发语言·qt
reddingtons3 小时前
在 Ubuntu 下通过 Docker 部署 Nginx 服务器
服务器·ubuntu·docker
大小科圣3 小时前
nginx实现tomcat反向代理
服务器·nginx·tomcat
放羊郎4 小时前
Ubuntu从源代码编译安装QT
linux·qt·ubuntu
杰克崔4 小时前
关于stac和clac的进一步细节及EFLAGS
linux·运维·服务器