shell中的fork、source和exec总结(包括环境变量)

来源:互联网 发布:rdpac 数据 编辑:程序博客网 时间:2024/05/29 16:31

摘要:对fork,source和exec三种方式执行shell脚本的总结。

 

准备知识

1.我们所执行的任何程序,都是由父进程(parent process)所产生出来的一个子进程(child process),子进程在结束后,将返回到父进程去。此一现像在Linux系统中被称为 fork。当子进程被产生的时候,将会从父进程那里获得一定的资源分配、及(更重要的是)继承父进程的环境﹗
2.Shell变量大致可以分为3种类型:
   内部变量:系统提供,不用定义,不能修改,比如$#,$?,$*,$0等
   环境变量:系统提供,不用定义,可以修改,当前进程及其子进程中使用,比如PATH,PWD,SHELL等
   用户变量(本地变量):用户定义,可以修改,在当前进程使用,比如var=123等
   与其他语言的区别:非类型性质,也就是不必指定变量是数字或字符串等。
3.关于环境变量:环境变量只能从父进程到子进程单向继承。换句话说:在子进程中的环境如何变更,均不会影响父进程的环境。
4.Shell脚本:其实就是将你平时在Shell prompt后所输入的多行command依序写入一个文件去而已

 

fork,source和exec方式执行Shell脚本

fork方式
也就是常用的方式,一般在shell直接输入脚本文件路径就可以了。这种方式由当前进程创建一个子进程

1:  ./mytest.sh


source方式
使用方式如下(source与".”等价):

1:  source ./mytest.sh
2:  或者
3:  . ./mytest.sh
.codearea{ color:black; background-color:white; line-height:18px; border:1px solid #4f81bd; margin:0; width:auto !important; width:100%; overflow:auto; text-align:left; font-size:12px; font-family: "Courier New","Consolas","Fixedsys","BitStream Vera Sans Mono", courier,monospace,serif}.codearea pre{ color:black; line-height:18px; padding:0 0 0 12px !important; margin:0em; background-color:#fff !important}.linewrap pre{white-space:pre-wrap; white-space:-moz-pre-wrap; white-space:-pre-wrap; white-space:-o-pre-wrap; word-wrap:break-word; word-break:normal}.codearea pre.alt{ background-color:#f7f7ff !important}.codearea .lnum{color:#4f81bd;line-height:18px}

source方式的特点是,在不另外创建子进程,而是在当前的的Shell环境中执行。

exec方式

1:  exec ./mytest.sh
.codearea{ color:black; background-color:white; line-height:18px; border:1px solid #4f81bd; margin:0; width:auto !important; width:100%; overflow:auto; text-align:left; font-size:12px; font-family: "Courier New","Consolas","Fixedsys","BitStream Vera Sans Mono", courier,monospace,serif}.codearea pre{ color:black; line-height:18px; padding:0 0 0 12px !important; margin:0em; background-color:#fff !important}.linewrap pre{white-space:pre-wrap; white-space:-moz-pre-wrap; white-space:-pre-wrap; white-space:-o-pre-wrap; word-wrap:break-word; word-break:normal}.codearea pre.alt{ background-color:#f7f7ff !important}.codearea .lnum{color:#4f81bd;line-height:18px}

此方式的特点是,不另外创建子进程,但是会终止当前的shell执行(其实我觉得这样理解可能更准确:使用exec会在当前的进程空间创建一个子线程,然后终止当前线程的执行,到了新建的线程执行完之后,其实两个线程都终止了,也就是这个当前shell进程也就终止了)

测试一
创建test1.sh,内容如下:

1:  #!/bin/sh
2:  cd ~/bin
3:  pwd

分别使用三种方式执行,source与exec方式都会将目录更改应用到当前环境,不同的是,exec方式执行完之后,shell环境就不能够使用了,会自动重启一个新的shell环境(进程)

测试二
一个更为详实的测试
脚本1.sh

 1:  #!/bin/sh
 2:  A=B
 3:  echo "PID for 1.sh before exec/source/fork:$$"
 4:  export A
 5:  echo "1.sh: /$A is $A"
 6:  case $1 in
 7:          exec)
 8:                  echo "using exec..."
 9:                  exec ./2.sh ;;
10:          source)
11:                  echo "using source..."
12:                  . ./2.sh ;;
13:          *)
14:                  echo "using fork by default..."
15:                  ./2.sh ;;
16:  esac
17:  echo "PID for 1.sh after exec/source/fork:$$"
18:  echo "1.sh: /$A is $A"
.codearea{ color:black; background-color:white; line-height:18px; border:1px solid #4f81bd; margin:0; width:auto !important; width:100%; overflow:auto; text-align:left; font-size:12px; font-family: "Courier New","Consolas","Fixedsys","BitStream Vera Sans Mono", courier,monospace,serif}.codearea pre{ color:black; line-height:18px; padding:0 0 0 12px !important; margin:0em; background-color:#fff !important}.linewrap pre{white-space:pre-wrap; white-space:-moz-pre-wrap; white-space:-pre-wrap; white-space:-o-pre-wrap; word-wrap:break-word; word-break:normal}.codearea pre.alt{ background-color:#f7f7ff !important}.codearea .lnum{color:#4f81bd;line-height:18px}

 脚本2.sh

1:  #!/bin/sh
2:  echo "PID for 2.sh: $$"
3:  echo "2.sh get /$A=$A from 1.sh"
4:  A=C
5:  export A
6:  echo "2.sh: /$A is $A"
.codearea{ color:black; background-color:white; line-height:18px; border:1px solid #4f81bd; margin:0; width:auto !important; width:100%; overflow:auto; text-align:left; font-size:12px; font-family: "Courier New","Consolas","Fixedsys","BitStream Vera Sans Mono", courier,monospace,serif}.codearea pre{ color:black; line-height:18px; padding:0 0 0 12px !important; margin:0em; background-color:#fff !important}.linewrap pre{white-space:pre-wrap; white-space:-moz-pre-wrap; white-space:-pre-wrap; white-space:-o-pre-wrap; word-wrap:break-word; word-break:normal}.codearea pre.alt{ background-color:#f7f7ff !important}.codearea .lnum{color:#4f81bd;line-height:18px}

分别使用三种方式执行1.sh脚本,结果如下:
~$ ./1.sh fork

1:  PID for 1.sh before exec/source/fork:531
2:  1.sh: $A is B
3:  using fork by default...
4:  PID for 2.sh:532
5:  2.sh get $A=B from 1.sh
6:  2.sh: $A is C
7:  PID for 1.sh after exec/source/fork:531
8:  1.sh: $A is B
.codearea{ color:black; background-color:white; line-height:18px; border:1px solid #4f81bd; margin:0; width:auto !important; width:100%; overflow:auto; text-align:left; font-size:12px; font-family: "Courier New","Consolas","Fixedsys","BitStream Vera Sans Mono", courier,monospace,serif}.codearea pre{ color:black; line-height:18px; padding:0 0 0 12px !important; margin:0em; background-color:#fff !important}.linewrap pre{white-space:pre-wrap; white-space:-moz-pre-wrap; white-space:-pre-wrap; white-space:-o-pre-wrap; word-wrap:break-word; word-break:normal}.codearea pre.alt{ background-color:#f7f7ff !important}.codearea .lnum{color:#4f81bd;line-height:18px}

~$ ./1.sh source

1:  PID for 1.sh before exec/source/fork:533
2:  1.sh: $A is B
3:  using source...
4:  PID for 2.sh:533
5:  2.sh get $A=B from 1.sh
6:  2.sh: $A is C
7:  PID for 1.sh after exec/source/fork:533
8:  1.sh: $A is C
.codearea{ color:black; background-color:white; line-height:18px; border:1px solid #4f81bd; margin:0; width:auto !important; width:100%; overflow:auto; text-align:left; font-size:12px; font-family: "Courier New","Consolas","Fixedsys","BitStream Vera Sans Mono", courier,monospace,serif}.codearea pre{ color:black; line-height:18px; padding:0 0 0 12px !important; margin:0em; background-color:#fff !important}.linewrap pre{white-space:pre-wrap; white-space:-moz-pre-wrap; white-space:-pre-wrap; white-space:-o-pre-wrap; word-wrap:break-word; word-break:normal}.codearea pre.alt{ background-color:#f7f7ff !important}.codearea .lnum{color:#4f81bd;line-height:18px}

~$ ./1.sh exec

1:  PID for 1.sh before exec/source/fork:537
2:  1.sh: $A is B
3:  using exec...
4:  PID for 2.sh:537
5:  2.sh get $A=B from 1.sh
6:  2.sh: $A is C
.codearea{ color:black; background-color:white; line-height:18px; border:1px solid #4f81bd; margin:0; width:auto !important; width:100%; overflow:auto; text-align:left; font-size:12px; font-family: "Courier New","Consolas","Fixedsys","BitStream Vera Sans Mono", courier,monospace,serif}.codearea pre{ color:black; line-height:18px; padding:0 0 0 12px !important; margin:0em; background-color:#fff !important}.linewrap pre{white-space:pre-wrap; white-space:-moz-pre-wrap; white-space:-pre-wrap; white-space:-o-pre-wrap; word-wrap:break-word; word-break:normal}.codearea pre.alt{ background-color:#f7f7ff !important}.codearea .lnum{color:#4f81bd;line-height:18px}

 注意:使用exec执行时1.sh中最后两句并没有被执行

参考资料:

http://gaoyj1973.itpub.net/post/11309/498996
http://hi.baidu.com/cn_linux/blog/item/923cad8bf46cac7e9e2fb49d.html
http://blog.chinaunix.net/u2/63996/showart_573729.html
http://hi.baidu.com/252568175/blog/item/f3489918d585a80b34fa4143.html
原创粉丝点击