Chinaunix

标题: 进程停止和启动问题 [打印本页]

作者: chen_hy    时间: 2009-03-16 10:01
标题: 进程停止和启动问题
各位大虾.小弟最近遇到了一个问题,请指教.
我写了一个shell脚本stop.sh来停止进程aa,代码如下:

#!/bin/sh
  tmpfile=/tmp/aaProcessID
  ps -e|grep aa|cut -c 1-8 > $tmpfile
  chmod 775 $tmpfile
  cnt=`wc -l $tmpfile | cut -c 1-8`
  if [ $cnt -gt 0 ]
  then
        for i in `cat $tmpfile`
        do
                kill -s USR1 $i
        done
  fi

  test -f $tmpfile
  if [ $? -eq 0 ]
  then
        rm -f $tmpfile
  fi

以上代码的基本思路是将aa进程的进程号写入tmpfile文件,然后读tmpfile文件将所有aa进程停止掉.
现在的问题是可能在停止aa进程之前需要做一些判断,例如有没有在进行读文件操作.如果有的话要等到读文件操作完成
之后再停止该进程.
我现在想写一个stop.c来实现以上功能.但是不知道unix c里面取得某个进程aa的进程号的函数是什么.
或者各位大虾有更好的想法,请指教.谢谢.
作者: sleetboy    时间: 2009-03-16 10:03
提示: 作者被禁止或删除 内容自动屏蔽
作者: chen_hy    时间: 2009-03-16 10:08
killall是停止所有进程,这样不好吧.呵呵.
作者: gawk    时间: 2009-03-16 10:29
用popen把ps -ef|grep xxx的东西读出来处理如何?
作者: chen_hy    时间: 2009-03-16 10:50
谢谢楼上的.我试试看.
作者: converse    时间: 2009-03-16 11:31
man pidof命令.
或者进程创建的时候将pid写入一个文件中.
作者: happy_fish100    时间: 2009-03-16 12:10
原帖由 chen_hy 于 2009-3-16 10:01 发表
现在的问题是可能在停止aa进程之前需要做一些判断,例如有没有在进行读文件操作.如果有的话要等到读文件操作完成
之后再停止该进程.
   ...


处理程序aa中通过sigaction或signal捕获SIGINT或SIGTERM信号,设置退出标记,然后程序正常退出。
BTW:读文件操作还需要等待进程操作完成?我觉得写文件操作才需要等待操作完成吧?
作者: chen_hy    时间: 2009-03-17 10:08
谢谢各位对我的问题的关注和支持.现在把我的想法说一下,请大家指教.

先说一下我的需求:
我写了一个守护进程,每隔一个间隔时间(参数化)读取固定文件夹下的文件,然后做相应的处理.该进程是由startup.sh来启动的.另外停止进程是由stop.sh来完成的.
需求要求在停止进程的时候判定读取一个文件后相应的处理有没有完成.如果没有完成,需要等到它完成后再停止.

2楼的sleetboy建议用killall 进程名字,4楼的gawk建议用popen把ps -ef|grep xxx的东西读出来处理.这两种方法因为无法实现守护进程和停止进程之间的联系,
可能有一定的困难.7楼的happy_fish100建议通过sigaction或signal捕获SIGINT或SIGTERM信号.我觉得比较好.只是小弟昨天试了一下,好象没有成功.我不知道
是不是和守护进程有关.现在将代码贴上来,大家给提提建议.谢谢.

守护进程主程序aa.c:
void quit(int signo)
{      
        char strProcessID[128];
        memset(strProcessID, 0, sizeof(strProcessID));
        char strCmd[128];
        memset(strCmd, 0, sizeof(strCmd));
        sprintf(strCmd, "ps -e|grep aa|cut -c 1-8");
        FILE * fp = NULL;
        fp = popen(strCmd, "r");
        while(fgets(strProcessID, sizeof(strProcessID), fp))
        {
                printf("strProcessID=%s\n", strProcessID);
                memset(strCmd, 0, sizeof(strCmd));
                sprintf(strCmd, "kill -s USR1 %s", strProcessID);
                if (0 > system(strCmd))
                {
                        WriteLog("无法终止aa系统\n");
                        return -1;
                }
        }
        pclose(fp);
        return 0;
}

main()
{
                signal(SIGTTOU, SIG_IGN);
                signal(SIGTTIN, SIG_IGN);
                signal(SIGTSTP, SIG_IGN);
                signal(SIGHUP, SIG_IGN);

                 struct sigaction act;
                 act.sa_handler = quit;
                 act.sa_flags      = 0;
                 sigemptyset(&act.sa_mask);
                 sigaction(SIGTERM, &act, NULL);

                 if (0 != fork())
                 {
                             exit(1);  
                 }
                 if (0 > setsid())
                 {
                              exit(1);
                 }
                 if (0 != fork())
                 {
                                exit(1);
                 }
                  if (-1 == chdir("/tmp"))
                  {
                                 exit(1);
                  }
                   umask(0);
                   signal(SIGCHLD, SIG_IGN);

                    while(1)
                    {
                                         if (ERR_FILE_NOT_EXIST == aa_main())      /*文件夹下没有文件,sleep10秒*/
                                         {
                                                         sleep(10);
                                          }
                     }
}

startup.sh:
#!/bin/sh
  tmpfile=/tmp/aaProcessID
  ps -e|grep aa|cut -c 1-8 > $tmpfile
  chmod 775 $tmpfile
  cnt=`wc -l $tmpfile | cut -c 1-8`
  if [ $cnt -gt 0 ]
  then
        for i in `cat $tmpfile`
        do
                kill -s USR1 $i
        done
  fi

  test -f $tmpfile
  if [ $? -eq 0 ]
  then
        rm -f $tmpfile
  fi

  ./aa

stop.sh:
#!/bin/sh
  tmpfile=/tmp/aaProcessID
  ps -e|grep aa|cut -c 1-8 > $tmpfile
  chmod 775 $tmpfile
  cnt=`wc -l $tmpfile | cut -c 1-8`
  if [ $cnt -gt 0 ]
  then
        for i in `cat $tmpfile`
        do
                kill -s USR1 $i
        done
  fi

  test -f $tmpfile
  if [ $? -eq 0 ]
  then
        rm -f $tmpfile
  fi
作者: gawk    时间: 2009-03-17 10:40
#include <signal.h>

int kill(
         pid_t process,
         int signal );
楼主看看这个函数
作者: chen_hy    时间: 2009-03-17 10:47
回楼上的.我看了kill函数,是给进程发送一个信号.
作者: chen_hy    时间: 2009-03-17 15:23
各位大哥,给点意见啊.谢谢.
作者: chen_hy    时间: 2009-03-30 13:39
各位大虾.我已经用消息队列解决该问题.谢谢各位的指导.
作者: gawk    时间: 2009-03-30 13:44
原帖由 chen_hy 于 2009-3-30 13:39 发表
各位大虾.我已经用消息队列解决该问题.谢谢各位的指导.

放出来学习一下
作者: chen_hy    时间: 2009-04-01 11:36
标题: 回复 #13 gawk 的帖子
在aa_main()函数中接收消息队列的消息,如果有消息到来,sleep 20秒.
在stop.cc文件中发送消息队列,同时sleep10秒,等待aa_main处理完读文件之后的其他事情.之后停止进程.

各位大虾如果有好的想法,请多多指教啊.小弟谢谢了.




欢迎光临 Chinaunix (http://bbs.chinaunix.net/) Powered by Discuz! X3.2