1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
1
#!/bin/sh
2 a=main
3
4
echo
"a is $a"
5
echo
"PID for parent before 2.sh:$$"
6
case
$1
in
7
exec
)
8
echo
"using exec"
9
exec
.
/2
.sh ;;
10
source
)
11
echo
"using sourcing"
12
source
.
/2
.sh ;;
13 *)
14
echo
"using fork"
15 .
/2
.sh ;;
16
17
esac
18
19
echo
"PID FOR parent after 2.sh :$$"
20
21
echo
"now main.sh a is $a"
22
echo
"$b"
|
1
2
3
4
5
6
7
8
9
10
11
12
|
1
#!/bin/sh
2
echo
"PID FOR 2.SH:$$"
3
4
echo
"2.sh get a from main.sh is $a"
5
6 a=2.sh
7
export
a
8 b=3.sh
9
10
echo
"now 2.sh a is $a"
~
~
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
[root@lvs2 home]
# ./main.sh exec
a is main
PID
for
parent before 2.sh:19026
using
exec
PID FOR 2.SH:19026
2.sh get a from main.sh is main
now a is 2.sh
[root@lvs2 home]
# ./main.sh source
a is main
PID
for
parent before 2.sh:19027
using sourcing
PID FOR 2.SH:19027
2.sh get a from main.sh is main
now a is 2.sh
PID FOR parent after 2.sh :19027
now main.sh a is 2.sh
3.sh
[root@lvs2 home]
# ./main.sh fork
a is main
PID
for
parent before 2.sh:19028
using fork
PID FOR 2.SH:19029
2.sh get a from main.sh is main
now a is 2.sh
PID FOR parent after 2.sh :19028
now main.sh a is main
[root@lvs2 home]
#
|
==================================================================
原文地址:http://blog.youkuaiyun.com/wfq_1985/article/details/5429838
==================================================================
学过C语言的都知道,Unix下某个进程的内存分成三部分:代码段,堆栈段,数据段。代码段用来存放程序运行的代码,堆栈段用来存放子程序的局部变量,数据段用来存放全局变量。这在perl里也是一样的。
perl的fork调用,跟C的一样,当发生fork调用时,实际上发生如下事:
父进程将代码段,堆栈段,数据段完全复制一份给子进程。也就是说,在子进程运行之初,它拥有父进程的一切变量和句柄。例如,父进程申明了某个hash表,那这个hash表也会被子进程拥有。
然而,一旦子进程开始运行,它的数据段和堆栈段就在内存里完全和父进程分离开了。也就是说,两个进程间不再共享任何数据。例如前面所说的hash表,虽然子进程从父进程处继承了这个数据结构,但子进程写往hash里的数据,不会被父进程访问到。在shell里用ps命令,可以看到2个独立运行的进程。通常你 kill掉1个,不会影响另1个的运行。
那么父进程和fork出来的子进程如何通信呢?父进程和子进程间的通信有多种方法,最常见的是信号,另外还有管道,Socket,消息队列等,不在这里详叙。而2个进程间共享数据的办法,可以用线程或共享内存,我对这方面不熟悉。
如果大概明白了fork,那么exec就容易理解了。一个进程一旦调用exec类函数,它本身就“死亡”了,系统把代码段替换成新的程序的代码,废弃原有的数据段和堆栈段,并为新程序分配新的数据段与堆栈段,唯一留下的,就是进程号,也就是说,对系统而言,还是同一个进程,不过已经是另一个程序了。
在perl里,调用exec后,原进程就完全消失,由于消失了,它也就不会从新进程接受到任何返回值,除非新进程意外终止,原进程会接受到错误值。