Chinaunix首页 | 论坛 | 博客
  • 博客访问: 21999
  • 博文数量: 13
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 10
  • 用 户 组: 普通用户
  • 注册时间: 2013-08-28 20:53
个人简介

我喜欢虚拟的世界!

文章分类
文章存档

2013年(13)

我的朋友

分类: LINUX

2013-09-18 14:32:10

原文地址:进程详解 作者:caojiangfeng

一:相关概念

进程是一个非常重要的概念,许多的程序都会用子进程.创建一个子进程是每一个程序员的基本要求!

1:什么是程序,什么是进程呢?

通俗的讲程序是一个包含可以执行代码的文件,是一个静态的文件.而进程是一个开始执行但是还没有结束的程序的实例.就是可执行文件的具体实现. 一个程序可能有许多进程,而每一个进程又可以有许多子进程.依次循环下去,而产生子孙进程. 当程序被系统调用到内存以后,系统会给程序分配一定的资源(内存,设备等等)然后进行一系列的复杂操作,使程序变成进程以供系统调用.在系统里面只
有进程没有程序,为了区分各个不同的进程,系统给每一个进程分配了一个ID(就象我们的身份证)以便识别. 为了充分的利用资源,系统还对进程区分了不同的状态.将进程分为新建,运行,阻塞,就绪和完成五个状态. 新建表示进程正在被创建,运行是进程正在运行,阻塞是进程正在等待某一个事件发生,就绪是表示系统正在等待CPU 来执行命令,而完成表示进程已经结束了系统正在回收资源.

2:进程ID(#include ;下面的ID都是长整型,所以打印是ld格式)

系统调用getpid 可以得到进程的ID,而getppid 可以得到父进程(创建调用该函数进程的进程)的ID.

#include ;
pid_t getpid(void);
pid_t getppid(void);

进程是为程序服务的,而程序是为了用户服务的.系统为了找到进程的用户名,还为进程和
用户建立联系.
这个用户称为进程的所有者.相应的每一个用户也有一个用户ID.通过系统
调用getuid 可以得到进程的所有者的ID.由于进程要用到一些资源,而Linux 对系统资源是
进行保护的,为了获取一定资源进程还有一个有效用户ID.这个ID 和系统的资源使用有关
,涉及到进程的权限. 通过系统调用geteuid 我们可以得到进程的有效用户ID. 和用户ID
相对应进程还有一个组ID 和有效组ID 系统调用getgid 和getegid 可以分别得到组ID 和有效
组ID
#include ;
#include ;
uid_t getuid(void);
uid_t geteuid(void);
gid_t getgid(void);
gid_t getegid(void);
有时候我们还会对
用户的其他信息感兴趣(登录名等等),这个时候我们可以调用getpwuid 来得到.

#include ;
#include ;
struct passwd *getpwuid(uid_t uid);

struct passwd {
char *pw_name;
char *pw_passwd;
uid_t pw_uid;
gid_t pw_gid;
char *pw_gecos;
char *pw_dir;
char *pw_shell;
};

3:实例(进程的相关信息)

#include ;
#include ;
#include ;
#include ;
int main(int argc,char **argv)
{
pid_t my_pid,parent_pid;
uid_t my_uid,my_euid;
gid_t my_gid,my_egid;
struct passwd *my_info;
my_pid=getpid();
parent_pid=getppid();
my_uid=getuid();
my_euid=geteuid();
my_gid=getgid();
my_egid=getegid();
my_info=getpwuid(my_uid);
printf("Process ID:%ld\n",my_pid);
printf("Parent ID:%ld\n",parent_pid);
printf("User ID:%ld\n",my_uid);
printf("Effective User ID:%ld\n",my_euid);
printf("Group ID:%ld\n",my_gid);
printf("Effective Group ID:%ld\n",my_egid):
if(my_info)
 
   {
printf("My Login Name:%s\n" ,my_info->;pw_name);
printf("My Password :%s\n" ,my_info->;pw_passwd);
printf("My User ID :%ld\n",my_info->;pw_uid);
printf("My Group ID :%ld\n",my_info->;pw_gid);
printf("My Real Name:%s\n" ,my_info->;pw_gecos);

printf("My Home Dir :%s\n", my_info->;pw_dir);
printf("My Work Shell:%s\n", my_info->;pw_shell);
 
   }
}

二:进程的创建(一定要会)

#include ;
pid_t fork();
当一个进程调用了fork 以后,系统会创建一个子进程.这个子进程和父进程不同的地方只
有他的进程ID 和父进程ID,其他的都是一样.

返回值:

当fork 调用失败的时候(内存不足或者是用户的最大进程数已到)fork 返回-1,否则fork 的返回值有重要的作用.对于父进程fork 返回子进程的ID,而对于fork 子进程返回0.我们就是根据这个返回值来区分父子进程的.

父进程为什么要创建子进程呢?前面我们已经
说过了Linux 是一个多用户操作系统,在同一时间会有许多的用户在争夺系统的资源.
有时
进程为了早一点完成任务就创建子进程来争夺资源.
一旦子进程被创建,父子进程一起从
fork 处继续执行,相互竞争系统的资源.
有时候我们希望子进程继续执行,而父进程阻塞直
到子进程完成任务.这个时候我们可以调用
wait 或者waitpid 系统调用.
#include ;
#include ;
pid_t wait(int *stat_loc);//等待子进程结束或者等待接收信号
pid_t waitpid(pid_t pid,int *stat_loc,int options);//等待指定的子进程结束
wait 系统调用会使父进程阻塞直到一个子进程结束或者是父进程接受到了一个信号.如果
没有父进程没有子进程或者他的子进程已经结束了wait 会立即返回.
败返回-1,并设置全局变量errno;成功时(因一个子进程结束)wait 将返回子进程的ID,stat_loc 是子进程的退出状态.子进程调用exit,_exit 或者是return 来设置这个值. 为了得到这个值Linux 定义了几个宏来测试这个返回值.
WIFEXITED(stat_loc);//:判断
子进程退出值是非0
WEXITSTATUS(stat_loc);//:判断子进程的退出值(当子进程退出时非0时).
WIFSIGNALED(stat_loc);//:子进程由于有没有获得的信号而退出.
WTERMSIG(stat_loc);//:
子进程没有获得的信号(在WIFSIGNALED 为真时才有意义).
waitpid 等待指定的子进程直到子进程返回.如果pid 为正值则等待指定的进程(pid).如果
为0 则等待任何一个
同组ID的进程.为-1 时等同于wait 调用.小于-1 时等
待任何一个
组ID 等于pid 绝对值的进程. stat_loc 和wait 的意义一样. options 可以决定
父进程的状态.可以取两个值

WNOHANG:父进程立即返回当没有子进程存在时.
WUNTACHED:当子进程结束时waitpid 返回,但是子进程的退出状态不可得到.
父进程创建子进程后,子进程一般要执行不同的程序.为了调用系统程序,我们可以使用系统调用exec 族调用.exec 族调用有着5 个函数.
#include ;
int execl(const char *path,const char *arg,...);
int execlp(const char *file,const char *arg,...);
int execle(const char *path,const char *arg,...);
int execv(const char *path,char *const argv[]);
int execvp(const char *file,char *const argv[]):
exec 族调用可以执行给定程序.关于exec 族调用的详细解说可以参考系统手册(man execl).

实例:注意编译的时候要加 -lm 以便连接数学函数库

#include ;
#include ;
#include ;
#include ;
#include ;
#include ;
void main(void)
{
pid_t child;
int status;
printf("This will demostrate how to get child status\n");
if((child=fork())==-1)
{
printf("Fork Error :%s\n",
strerror(errno));
exit(1);//创建子进程失败退出
}
else
if(child==0)
{
int i;
printf("I am the child
%ld\n",getpid());
for(i=0;i<1000000;i++)

sin(i);
i=5;
printf("I exit with %d\n",i);
exit(i);//我是子进程退出
}//是父进程而且创建子进程成功才进行下列操作,否则退出
while(((child=wait(&status))==-1)&(errno==EINTR));
if(child==-1)
printf("Wait Error:%s\n",strerror(errno));
else if(!status)
printf("Child %ld terminated normally return status is zero\n",
child);
else if(WIFEXITED(status))
printf("Child %ld terminated normally return status is
%d\n",
child,WEXITSTATUS(status));
else if(WIFSIGNALED(status))

printf("Child %ld terminated due to signal %d not caught\n",
child,WTERMSIG(status));
}
strerror 函数会返回一个指定的错误号的错误信息的字符串.

三:守护进程的创建

 如果你在DOS 时代编写过程序,那么你也许知道在DOS 下为了编写一个常驻内存的
程序我们要编写多少代码了.相反如果在Linux 下编写一个
"常驻内存"的程序却是很容易的.
我们只要几行代码就可以做到. 实际上
由于Linux 是多任务操作系统,我们就是不编写代码
也可以把一个程序放到后台去执行的.我们只要在命令后面加上&符号SHELL 就会把我们的
程序放到后台去运行的.
这里我们"开发"一个后台检查邮件的程序.这个程序每个一个指
定的时间回去检查我们的邮箱,如果发现我们有邮件了,会不断的报警(通过机箱上的小喇
叭来发出声音). 后面有这个函数的加强版本,
后台进程的创建思想: 首先父进程创建一个子进程.然后子进程杀死父进程(是不是很无情?). 信号处理所有的工作由子进程来处理.
#include ;
#include ;
#include ;
#include ;
#include ;
#include ;
#include ;

#define MAIL "/var/spool/mail/hoyt"

#define SLEEP_TIME 10
main(void)
{
pid_t child;
if((child=fork())==-1)
 
   {
printf("Fork Error:%s\n",strerror(errno));
exit(1);
 
   }
else if(child>0)
while(1)
    if(kill(getppid(),SIGTERM)==-1)
 
   {
printf("Kill Parent Error:%s\n",strerror(errno));
exit(1);

    }
 
   {
int mailfd;
while(1)
 
      {
if((mailfd=open(MAIL,O_RDONLY))!=-1)
 
        {
fprintf(stderr,"%s","\007");
close(mailfd);
 
         }
sleep(SLEEP_TIME);
 
      }
 
    }
}

你可以在默认的路径下创建你的邮箱文件,然后测试一下这个程序.当然这个程序还有很
多地方要改善的.我们后面会对这个小程序改善的,再看我的改善之前你可以尝试自己改
善一下.比如让用户指定邮相的路径和睡眠时间等等.相信自己可以做到的.

这篇文章是我在学习完进程之后在网上看到的,感觉一下将我所学的知识串了起来,一下自明白了学习完进程之后,它的知识点之间的关系一下清晰了起来。
阅读(592) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~