复习:
1.什么是进程
动态
2.僵尸进程
子进程消亡 父进程没有消亡 但不为子进程收尸 此时子进程是僵尸进程 避免僵尸进程
如何避免僵尸进程:
1)收尸函数
wait
waitpid
2)子进程消亡通知父进程 父进程收尸
3)kill 父
3.孤儿进程
父进程消亡 子进程在运行 子进程是孤儿进程 init领养并收尸 PID 1
4.fork功能
功能:创建子进程
id = fork()
if(id<0)
{
}
else if(id==0)//子进程
{
}
else //id>0 父进程
{
}
5.收尸函数有哪些
6.exit _exit区别
exit 退出 刷新缓存
_exit 立即退出 不刷新缓存
作业:
练习: 子进程每秒打印一个数,一共打印 10 个,执行完毕,父进程收尸。也尝试一次不阻塞的情形。
#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
int main()
{
//创建子进程
pid_t pid = fork();
if( pid < 0)
{
perror(“fork”);
return -1;
}
else if(pid > 0) // 父进程
{
//为任意子进程阻塞收尸
waitpid(-1,NULL,0);//waitpid(-1,NULL,WNOHANG)
printf(“OVER!\n”);
}
else
{
int i;
for(i=0; i<10; i++)
{
printf("%d \n",i+1);
sleep(1);
}
}
}
练习: 编程实现 执行ls -l 执行完毕后 打印finished
子进程 ls -l
父 收尸并打印
#include<unistd.h>
#include<stdio.h>
#include<sys/types.h>
int main()
{
pid_t pid = fork();
if(pid<0)
{
perror(“fork”);
}
else if(pid>0)//父
{
//收尸
wait(NULL);
puts(“finished!”);
}
else //子
{
//执行任务
execl("/bin/ls",“ls”,"-l",NULL);
}
}
一、进程执行任务续
int execvp(const char *file, char *const argv[]);
格式:execvp(可执行文件名,指针数组)
实例1:
#include"my.h"
int main()
{
puts(“come…”);
//execl("/bin/ls","ls","-l",NULL);
char *cmd[]={"ls","-l",NULL};
execvp("ls",cmd);//ls -l
puts("bye....");
}
示例2:自定义任务
二、守护进程编写(精灵)
前台进程:依附终端 终端结束 进程结束
后台进程:不依附终端 自己独立存在
进程组:多个进程
会话:多个进程组
1.创建子进程 父进程结束(init领养子进程)
2.创建新会话 setsid()
3.改变当前工作目录
chdir("/tmp")
4.修改文件掩码
掩码:掩掉权限
umask(0)
5.关闭所有打开的文件描述符
int getdtablesize(void);//获取当前进程打开的文件总个数
int n=getdtablesize();//5
for(i=0;i<n;i++)
{
close(i);
}
文件描述符分配规则:
当前尚未分配最小非负 0 1 2 3 4
实例:获取计算机时间
#include<time.h>
#include<stdio.h>
int main()
{
time_t t = time(NULL);//获取1970.1.1到现在经历的秒数
char *p = ctime(&t);//将时间转为字符串时间
printf("%s\n",p);
return 0;
}
fclose 关闭给定的文件流。任何未写入的缓冲数据都会被刷新到OS中。任何未读的缓冲数据都会被丢弃。
例子:
#include"my.h"
int main()
{
FILE *fp=NULL;
time_t t;
//1.创建守护进程
//(1)创建子进程 父进程退出
pid_t id = fork();
if(id>0)
{
exit(0);
}
只剩下子进程
//(2)创建会话
setsid();
//(3)修改当前工作目录
chdir("/tmp");
//(4)修改掩码
umask(0);
//(5)关闭所有打开的文件描述符号
int n = getdtablesize();
int i;
for(i=0;i<n;i++)
{
close(i);
}
//2.创建日志文件并写入数据
while(1)
{
fp = fopen("time.log","a+");
if(fp!=NULL)
{
//获取系统时间
t = time(NULL);
//将ctime获取的字符串系统时间 输出到文件fp中
fprintf(fp,"%s",ctime(&t));
fclose(fp);
sleep(1);
}
}
}
///my.h
#ifndef _MY_H
#define _MY_H
#include<stdio.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<unistd.h>
#include<strings.h>
#include<dirent.h>
#include<sys/wait.h>
#include<stdlib.h>
#include<time.h>
#endif
练习:编写一个守护进程 每隔1s向文件中写入一个字母 字母从’a’开始 ‘a’~‘z’
#include<stdio.h>
#include<sys/types.h>
#include<unistd.h>
#include<stdlib.h>
void init_daemon()
{
pid_t pid = fork();
if(pid>0)
{
exit(0);
}
setsid();
chdir("/tmp");
umask(0);
int n = getdtablesize();
int i;
for(i=0;i<n;i++)
{
close(i);
}
}
int main()
{
init_daemon();//创建守护进程
FILE *fp=NULL;
time_t t;
char ch='a';
while(1)
{
fp = fopen("char.txt","a");
if(fp!=NULL)
{
//将变量ch的值写入文件fp
fprintf(fp,"%c",ch);
fclose(fp);
if(ch=='z')
{
ch='a';
continue;
}
ch++;
sleep(1);
}
}
}