实验六 进程基础

项目 内容
这个作业属于的课程 课程班级的主页链接
这个作业要求在哪里 作业要求的链接地址
学号-姓名 18043228-章宇翔
作业学习目标 1.掌握Linux系统环境C语言编程概念 2.学习Linux系统进程概念

1请举例说明静态链接库的创建与使用

静态函数库实际上就是简单的一个普通的目标文件的集合,一般来说习惯用“.a”作为文件的后缀。可以用ar这个程序来产生静态函数库文件。Ar 是archiver的缩写。静态函数库现在已经不在像以前用得那么多了,主要是共享函数库与之相比较有很多的优势的原因。慢慢地,大家都喜欢使用共享函数库了。不过,在一些场所静态函数库仍然在使用,一来是保持一些与以前某些程序的兼容,二来它描述起来也比较简单。

 静态库函数允许程序员把程序link起来而不用重新编译代码,节省了重新编译代码的时间。

int add(int a,int b){
return a+b;
}
int mutiply(int a, int b) {  
    return a * b;
}
#include <stdio.h>
int add(int a,int b);
int mutiply(int a,int b);
int main(){
printf("1+1=%d\n",add(1,1));
printf("2*2=%d\n",mutiply(2,2));
return 0;
}

实验六 进程基础

 

 实验六 进程基础

 

 实验六 进程基础

gcc -c -static add.c -o add.o #将add.c编译成目标文件add.o
gcc -c -static mutiply.c -o mutiply.o 
ar -r libbase.a add.o #将目标文件生成libbase.a静态库
ar -r libbase.a mutiply.o 
#ar -r libbase.a add.o mutiply.o #上面的两条命令可以合并成一条 
ar -t libbase.a #可以看到包里所有的 .o 文件

实验六 进程基础

2请举例说明共享库的创建与使用 开始的目录结构 实验六 进程基础   实验六 进程基础

 

 创建自己的共享库

  

gcc -c -fpic add.c sub.c #编译 -fpic生成位置无关的代码
gcc -shared add.o sub.o -o ../lib/libbase.so #链接生成lib目录下的libbase.so共享库
#上面的命令执行完成后,会在 lib 目录下生成 libbase.so 文件 
#上面两条命令也可以用下面的一条命令 
#gcc -fpic -shared add.c sub.c -o ../lib/libbase.so

 

使用自己的共享库

gcc -c main.c -I../include #-I指编译程序按照指定的路进去搜索头文件
 gcc main.o ../lib/libbase.so -o ../bin/app #使用共享库将链接生成的可执行文件放于bin下 
 #可以使用ldd查看共享库 
 ldd bin/app #ldd列出动态库依赖关系

实验六 进程基础

 实验六 进程基础

 

 

3编程实现一个简单文件复制命令 实验六 进程基础

 实验六 进程基础

./mycp mycp.c test #文件复制
diff mycp.c test #diff以逐行的方式,比较文本文件的异同处

 实验六 进程基础

4使用fork创建一个子进程,进程创建成功后父子进程分别输出不同的内容。 实验六 进程基础

实验六 进程基础

删除fork1.c文件中 fflush(NULL); 这一行后运行结果为

 实验六 进程基础

 

继续删除fork1.c文件中 “ printf("[%d]:Begin! \n",getpid()); ” 这一句中的“\n”结果为:

 实验六 进程基础

5使用fork创建多个子进程。  实验六 进程基础

 实验六 进程基础

6在 fork 之前以写的方式创建了一个文件 test.txt。然后 fork 出的子进程立即向文件中写入“world”,然后睡眠5秒。而父进程在 fork 后睡眠3秒后向 test.txt 写入 "hello",并关闭描述符。子进程恢复后,又向 test.txt 文件中写入 "lalala"后关闭描述符,结束
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
int main() {
    int fd = open("test.txt",O_WRONLY | O_CREAT,0664);
    if (fd == -1){
        perror("open");
        return 1;
    }
    printf("I'm father\n");
    printf("before fork\n");
    pid_t pid = fork();
    if (pid > 0){
    sleep(3);
    printf("I'm father; I'm writing test.txt...\n");
    write(fd, "hello", 5);
    close(fd);
    }
    else if (pid ==0){
    printf("I'm child; I'm writing test.txt...\n");
    write(fd, "world", 5);
    sleep(5);
    write(fd, "lalala", 6);
    close(fd);
    }
    else {
        perror("fork");
        return 1;
    }
    return 0;
}

 

实验六 进程基础

 

 实验六 进程基础

7分别在主函数中使用execvp启动ls命令以及使用fork函数产生子进程调用execvp启动ls
1)在主函数中使用 execvp 启动 ls 命令 实验六 进程基础

 

 实验六 进程基础

 

 2)使用 fork 函数产生子进程调用 execvp 启动 ls

/*************************************************************************
    > File Name: forkexec.c
    > Author: Chrisyzx
    > Mail: 850096674.com 
    > Created Time: 2021年05月29日 星期六 11时13分34秒
 ************************************************************************/

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
int main(){
    char* argv[]={"ls","-l",NULL};
    pid_t pid=fork();
    if(pid>0){
        printf("I am dad\n");
    }
    else if(pid ==0){
        printf("I am kid\n");
        if(execvp("ls",argv)==-1){
            perror("exec");
            return 1;
        }
    }
    else {
        perror ("fork");
        return 1;
    }
return 0;
}

 

实验六 进程基础

 

 实验六 进程基础

8创建5个僵尸进程,并在终端通过ps axf命令查看僵尸进程信息 实验六 进程基础

 

 实验六 进程基础

另开一终端输入 ps axf 查看僵尸进程,显示如下:

 

实验六 进程基础

 

 

9通过wait来清理僵尸进程。 实验六 进程基础

 

 实验六 进程基础

 

 

  10父进程通过waitpid函数等待特定子进程结束,若该子进程不结束,父进程一直阻塞。 实验六 进程基础

实验六 进程基础

实验六 进程基础
上一篇:LINUX 多进程编程实例


下一篇:Java Fork/Join