shell实现Unix进程间信息交换的几种方法

使用命名管道实现进程间信息交换  

使用kill命令和trap语句实现进程间信息交换  

使用点命令“.”实现进程间信息交换  

使用export语句实现父进程对子进程的信息传递  

一、使用命名管道
  命名管道是一种先进先出(FIFO)的数据结构,它允许两个进程通过管道联接实现信息交换。
在Unix系统中,命名管道是一种特殊类型的文件,因此可以对命名管道进行读写操作;当然,同样
也会有读写和执行等权限的限制。  

  通过下面的命令可以创建一个命名管道:
  /etc/mknod pipe_name p  

  其中“pipe_name”是要创建的命名管道的名字,参数p 必须出现在命名管道名字之后。  

  命名管道文件被创建后,一些进程就可以不断地将信息写入命名管道文件里,而另一些进程也
可以不断地从命名管道文件中读取信息。对命名管道文件的读写操作是可以同时进行的。下面的例子
显示命名管道的工作过程。  

  进程A、B、C中运行的程序只是一条简单的echo命令,它们不断地把信息写入到命名管道文件
/tmp/pipe1中。与此同时,程序中的“read msg” 命令不断地从命名管道文件/tmp/pipe1中读取这些
信息,从而实现这些进程间的信息交换。  

  程序执行时,首先创建命名管道文件,此时程序处于等待状态,直到A、B、C进程中某一个进程往
命名管道中写入信息时,程序才继续往下执行。使用rm命令可以删除命名管道文件从而清除已设置的
命名管道。  

  下面是一个用于记录考勤的例子:  

  在主机上运行的程序/tmp/text产生命名管道/tmp/pipe1,并不断地从命名管道中读取信息送屏幕
上显示。  

/tmp/text程序:
if [ ! -p /tmp/pipe1 ]
then
  /etc/mknode /tmp/pipe1 p
fi
while :
do
  read msg
  if [ “$msg" = “" ]
  then
continue
  else
echo “$msg"
  fi
done < /tmp/pipe1 

  在终端上运行的是雇员签到程序/tmp/text1。每个雇员在任何一台终端上键入自己的名字或代码,
程序/tmp/text1将把这个名字连同当时的签到时间送入命名管道。  

/tmp/text1程序:
tty=‘who am I | awk ‘{print $2}’’
    while :
    do
      echo “Enter your name: \c" > /dev/$tty
      read name
      today=‘date’
      echo “$name\t$today"
      done > /tmp/pipe1 

  当雇员从终端上输入自己的姓名后,运行/tmp/text程序的主机将显示类似下面的结果:  

wang Thu Jan 28 09:29:26 BTJ 1999
he Thu Jan 28 09:29:26 BTJ 1999
cheng Thu Jan 28 09:30:26 BTJ 1999
zhang Thu Jan 28 09:31:26 BTJ 1999 

二、使用kill命令和trap语句
  在Unix系统中,当检测到一个异常的内部状态,或者硬件及外部设备发出请求,或者执行某些指令时,
将会向系统中的进程发出信号报告事件产生。当进程捕获到这些信号后,系统便转去执行预先设定的默认
程序,完成指定的动作;这些预先设定的默认程序称之为信号的系统陷阱。  

  在shell中,使用trap语句为信号设置新的陷阱。当shell 捕获到一个信号时(信号11除外,因为
shell本身要利用这个信号进行内存分配),它将这个信号传递给所有当前正在执行的程序
(父程序和子程序),并分别执行父程序和子程序中已设置的信号陷阱。一旦陷阱程序执行结束,便返
回中断点,继续执行原来的程序流程。  

    trap语句的基本格式:
trap command_list signal_list 

  command_list: 由一个或多个命令(或命令组)构成的命令列表。当命令列表中含有多个命令时要
用单引号或双引号括起来,并且各命令间要用分号隔开。  

  signal_list:由一个或多个信号值构成的信号列表,各信号值间要用空格分开。  

  在一个shell程序(父程序)中重新设置信号的陷阱并不改变被这个程序所调用的子程序中同名信号
的陷阱。同样,在子程序中设置的信号陷阱也不影响父程序中同名信号的陷阱。  

  shell在读取trap语句时,要扫描一次命令列表,以便设置陷阱。在捕获信号后,shell再次扫描命令
列表,执行已设置好的陷阱程序(命令或命令组)。因此,如果命令列表中含有变量置换或命令置换表达
式,shell在第一次扫描命令列表时就会用当前的变量值或命令结果置换这些表达式,使得在捕获到信号而
去执行陷阱程序时,陷阱程序已经不是原来设置的陷阱程序了。为了避免这种情况发生,使用单引号而不是
使用双引号把trap语句中含有变量置换或命令置换表达式的命令列表括起来;因为单引号可以消除所有字符
的特殊含义,这样避免了shell在第一次扫描时执行任何置换或替代操作,直到命令列表被执行时才进行置
换或替代。  

  向一个程序或进程传递信号方法很多,比如在程序执行时按下Ctrl+c键或Del键,将向程序传递一个
SIGINT信号,执行该信号的系统陷阱将终止程序执行。使用kill命令传递信号是shell语言编程中最为常用
的方法。  

  kill命令的基本格式是:
  kill [ - signal ] PID  

  通常kill命令用来终止一个进程。但如果使用了带有短划线“-”的信号作为参数时,kill命令就发送
该信号给PID指示的一个或多个进程,而不是终止进程。当trap语句捕获到这个信号后便执行设定的信号陷阱
程序,实现进程间的相互通讯。  

  下面的例子显示了程序master和slave1、slave2间如何利用信号机制实现相互通讯的。首先在后台运行
程序slave1和slave2,然后运行程序master。在文件/tmp/pro_list中记录了这三个程序的进程号。  

  程序slave1首先设置信号15的陷阱,然后把自己的当前进程写入文件/tmp/pro_list;在获得master进程
号后,进入循环状态。当接收到master发出的信号15时,执行陷阱程序,显示相关信息后,向master发出信
号15。  

  程序slave2执行情况与slave1相似。  

  程序master也是首先设置信号15的陷阱,然后把自己的当前进程写入文件/tmp/pro_list。在取得所有
slave程序进程号后,向这些slave程序发出信号15,然后进入循环等待。当接收到slave1或slave2发出的信
号15时,执行陷阱程序,显示相关信息,杀死所有slave进程,清空文件/tmp/pro_list,然后退出。  

    程序/tmp/slave1:
slave() {
  echo “slave1 has received sighal from master"
  echo “Request master to kill slave1 process"
  kill -15 $master_pid
}
trap slave 15
echo “slave1_pid $$" >> /tmp/pro_list
sleep 1
while :
do
  master_pid=‘awk ’$1 ~/master/
   {print $2}‘/tmp/pro_list’
  if [ “$master_pid" != “" ]
  then break
  fi
done
while :
do
  sleep 1
done 

    程序/tmp/slave2:
slave() {
  echo “slave2 has received sighal from master"
  echo “Request master to kill slave2 process"
  kill -15 $master_pid
}
trap slave 15
echo “slave2_pid $$" >> /tmp/pro_list
sleep 1
while :
do
  master_pid=‘awk ’$1 ~/master/
   {print $2}‘/tmp/pro_list’
  if [ “$master_pid" != “" ]
  then break
  fi
done
while :
do
  sleep 1
done 

    程序/tmp/master:
kill_slave() {
      echo “Master has received signals
            from slave processes"
      echo “End all slave processes"
      kill -9 $slave_list
>/tmp/pro_list
    exit 0
}
trap kill_slave 15
echo “master_pid $$" >> /tmp/pro_list
sleep 1
slave_list=‘awk ’$1 ~/slave/
     {print $2}‘/tmp/pro_list’
echo “Current slave processes are:"
echo “$slave_list"
kill -15 $slave_list
while :
do
  sleep 1
done 

    执行程序:
$ cd /tmp
$ ./slave1&
15638
$ ./slave2&
16831
$ ./master
Current slave processes are:
15638
16831
slave1 has received signal 15 from master
Request master to kill slave1 process
slave2 has received signal 15 from master
Request master to kill slave2 process
Master has received signals from slave processes
End all slave processes
15638 Killed
16831 Killed
$ 

三、使用点命令“.”
  “.”点命令是shell的一个内部命令,它从指定的shell 文件中读入所有命令语句并在当前进程
中执行。 因此当多个shell进程(父子进程或无关进程均可)共享一组变量值时,就可以将这些变量
赋值语句定义到一个shell文件里,并在需要这些变量值的程序中使用点语句来引用这个shell文件,
从而实现变量值共享(对这些变量值的修改仅涉及到这个shell文件)。但要注意的是,这个shell文
件不能包括含有位置参数的语句,即不能接受$1、$2等命令行参数。  

  下面是一个在超市中发布每日商品价格的示范程序片段。发布每日商品价格统一由程序/tmp/jiage
来执行,它为每种商品价格赋值,并把相应的赋值语句写入文件/tmp/jiagebiao中。在各终端上运行的
收款程序/tmp/shoukuan将读入文件 /tmp/jiagebiao中所有赋值语句并在当前进程中执行,从而获取在
程序/tmp/jiage中设定的价格。  

  价格设定程序/tmp/jiage:  

echo “Enter the price of chicken,
       duck and fish: \c"
read chicken duck fish
exec 3>/tmp/jiagebiao
echo “chicken_price=$chicken" >&3
echo “duck_price=$duck" >&3
echo “fish_price=$fish" >&3 

  执行/tmp/jiage程序后,文件/tmp/jiagebiao中将有如下内容:  

chicken_price=5.4
duck_price=2.5
fish_price=4.2 

  收款程序/tmp/shoukuan:  

. /tmp/jiagebiao
count=0
while :
do
echo “Enter the trade name and
      quantities or input q to sum: \c"
read trade$count quantity$count
eval a=\$trade$count
if [ “$a" = “q" ]
then if [ $count -gt 0 ]
then
count=‘expr $count - 1’
fi
break
fi
count=‘expr $count + 1 ’
done
echo “\n‘date’"
echo “trade name\tquantity\tsum"
while [ “$count" -ge 0 ]
do
      eval trade=“\${trade$count}"
      eval trade_price=“${trade}_price"
      eval danjia=\${$trade_price}
      eval quantity=“\${quantity$count}"
      sum=‘echo “scale=2; $danjia
          *$quantity"|bc’
      echo “$trade\t\t$quantity\t\t$sum"
      count=‘expr $count - 1 ’
done 

  在终端上执行程序/tmp/shoukuan将有如下显示:  

Enter the trade name and quantities
 or input q to sum: chicken 2
Enter the trade name and quantities
 or input q to sum: fish 3
Enter the trade name and quantities
 or input q to sum: duck 4
Enter the trade name and quantities
  or input q to sum: q 

Thu Jan 28 09:29:29 BJT 1999:
duck 4 10
fish 3 12.6
chicken 2 10.8 

四、使用export语句
  通常shell变量是局部变量,无论是通过赋值操作还是通过命令赋值,其变量值只在当前进程中
有效。但是经过export语句说明的shell变量就成为一个全局性变量,其变量名和变量值可以传递给
子进程及其后代进程。在子进程中可以修改这个变量的值,但并不影响这个变量在父进程中的取值。  

  下面的例子中,父进程(/tmp/text)将赋值后的变量 pro_name传递给子进程(/tmp/text_child),
在子进程中对变量pro_name所赋的新值并不影响父进程中该变量的值。  

    /tmp/text程序:
pro_name=“PARENT"
echo “The variable pro_name is
 $pro_name in parent process"
export pro_name
/tmp/text_child
echo “The variable pro_name is
 $pro_name after retund to parent process" 

    /tmp/text_child程序:
echo“The variable pro_name ($pro_name) is
      transmited to child process"
pro_name=“CHILD"
echo “To change the variable pro_name to
   $pro_name in child process"
    执行程序/tmp/text:
$ /tmp/text
The variable pro_name is PARENT
    in parent process
The variable pro_name (PARENT)
     is transmited to child process
To change the variable pro_name to CHILD
    in child process
The variable pro_name is PARENT
    after retund to parent process
$ 
时间: 2024-08-19 03:58:23

shell实现Unix进程间信息交换的几种方法的相关文章

Linux/UNIX进程间的通信(1)

进程间的通信(1) 进程间的通信IPC(InterProcessCommunication )主要有以下不同形式: 半双工管道和FIFO:全双工管道和命名全双工管道:消息队列,信号量和共享存储:套接字和STREAMS 管道 pipe函数 当从一个进程连接到另一个进程时,我们使用术语管道.我们通常是把一个进程的输出通过管道连接到另一个进程的输入. 管道是由调用pipe函数创建的: #include<unistd.h> int pipe(intpipefd[2]); 经由参数pipefd返回两个文

进程间共享句柄三种方式

windows核心编程第三章中,句柄的共享讲了三种方式 1.在创建句柄的时候,设置可继承,在创建进程时,让子进程继承句柄, 然后通过各种途径(如命令行参数.环境变量.发送消息)让子进程知道即可.2.复制句柄,得先知道目的进程,然后在目标进程中复制句柄, 最后通过各种途径(如发送消息)通知目标进程3.给句柄命名,进程间相互知道名字即可. 第3种在使用Mutex限制只运行一个实例时已使用,前2种没有例子,因此自己写了个实例. 运行后,点击创建进程,会创建一个文件句柄,并置为可继承,然后创建一个环境变

shell 脚本实现乘法口诀表的两种方法——shell与C语言

shell 脚本实现乘法口诀表的两种方法--shell与C语言 话不多说直接给出代码: 1 #!/bin/bash 2 if [ $# -eq 0 ] //用于判断输入的参数个数为0 3 then 4 echo "welcome you!" 5 echo "this is a test with 2 methods to output arbitrarily mux table!" 6 else 7 echo "sorry you input invlia

UNIX 进程间通讯(IPC)概念(Posix,System V IPC)

 IPC(Inter-Process Communication,进程间通讯)可以有三种信息共享方式(随文件系统,随内核,随共享内存).(当然这里虽然说是进程间通讯,其实也是可以和线程相通的). 相对的IPC的持续性(Persistence of IPC Object)也有三种: 随进程持续的(Process-Persistent IPC) IPC对象一直存在,直到最后拥有他的进程被关闭为止,典型的IPC有pipes(管道)和FIFOs(先进先出对象) 随内核持续的(Kernel-persist

进程间通讯的三种方式

# 进程间通讯 # 1.进程对列multiprocessing.Quere import multiprocessing, time def foo(q): time.sleep(1) print('son process', id(q)) q.put(123) q.put('alex') if __name__ == '__main__': q = multiprocessing.Queue() print('main process', id(q)) p = multiprocessing.

向其他进程注入代码的三种方法

http://huaidan.org/archives/838.html 原版地址: http://www.codeproject.com/threads/winspy.asp?df=100&forumid=16291&select=1025152&msg=1025152 pdf格式下载: http://netxfly.blogbus.com/files/1163491746.pdf 作者:Robert Kuster 翻译:袁晓辉([email protected]) 摘要:如何向

【Linux】shell编程之给变量赋值的五种方法

1.  Shell中的变量类型 Shell中的变量可以分为环境变量.位置变量.预定义的特殊变量以及用户自定义变量. <!--[if !supportLists]-->2.  <!--[endif]-->环境变量 Shell中的环境变量是一类Shell预定义变量,是用于设置系统运行环境的变量,环境变量由系统统一命名.部分系统变量的值由系统设定,部分环境变量的值可以由用户给定. 环境变量的名称由大写字母组成,常用的Shell环境变量如下所示: HOME: 用户主目录的全路径名,cd $

shell实例浅谈之三产生随机数七种方法

一.问题 Shell下有时需要使用随机数,在此总结产生随机数的方法.计算机产生的的只是"伪随机数",不会产生绝对的随机数(是一种理想随机数).伪随机数在大量重现时也并不一定保持唯一,但一个好的伪随机产生算法将可以产生一个非常长的不重复的序列. 二.随机数 1.生成随机数的七种方法 (1)通过内部系统变量($RANDOM) echo $RANDOM 生成0-32767之间的整数随机数,若超过5位可以加个固定10位整数,然后进行求余. 生成400000~500000的随机数: #!/bin

[转]去除inline-block元素间间距的N种方法

来自张鑫旭-鑫空间-鑫生活[http://www.zhangxinxu.com] 一.现象描述 真正意义上的inline-block水平呈现的元素间,换行显示或空格分隔的情况下会有间距,很简单的个例子: <input /> <input type="submit" /> 间距就来了~~ 我们使用CSS更改非inline-block水平元素为inline-block水平,也会有该问题: .space a { display: inline-block; paddi