项目 | 内容 |
---|---|
这个作业属于的课程 | 课程班级的主页链接 |
这个作业要求在哪里 | 作业要求的链接地址 |
学号-姓名 | 18043228-章宇翔 |
作业学习目标 | 1.掌握Linux系统环境C语言编程概念 2.学习Linux系统进程概念 |
静态函数库实际上就是简单的一个普通的目标文件的集合,一般来说习惯用“.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
函数等待特定子进程结束,若该子进程不结束,父进程一直阻塞。