Shell编程-11-子Shell和Shell嵌套

目录

  • 什么是子Shell
  • 子Shell产生的途径
  • Shell脚本调用模式

什么是子Shell

? ? 子Shell的概念其实是贯穿整个Shell的,如果想要更好的理解和写Shell脚本则必须要了解子Shell的相关知识。其概念如下所示:

子Shell本质就是从当前的Shell环境中打开一个新的Shell环境,而新开的Shell称之为子Shell(SubShell),相应的开启子Shell的环境称之为父Shell。子Shell和父Shell是子进程和父进程的关系,而这个进程则全部是bash进程。子Shell可以从父Shell中继承变量、命令全路径、文件描述符、当前工作目录等。在子Shell中常用的两个变量如下所示:

  • $BASH_SUBSHELL:查看从当前进程开始的子Shell层数
  • $BASHPID:查看当前所处BASH的PID
在Linux系统中,系统运行的程序基本都是从CentOS 6.x(init)或CentOS7.x(systemd)PID为1的进程)继承而来的,所有的程序都可以看作为init的子进程。
# CentOS 6.x
[[email protected] data]# pstree -hp
init(1)─┬─NetworkManager(3643)
        ├─Xvnc(22811)
        ├─abrtd(4760)
        ├─acpid(3755)
        ├─atd(4801)
        ├─auditd(3392)───{auditd}(3393)
        ├─automount(3849)─┬─{automount}(3850)
        │                 ├─{automount}(3851)
        │                 ├─{automount}(3854)
        │                 └─{automount}(3857)
# CentOS 7.x
[[email protected] ~]# pstree -hp
systemd(1)─┬─ModemManager(1051)─┬─{ModemManager}(1068)
           │                    └─{ModemManager}(1076)
           ├─Xvnc(5563)─┬─{Xvnc}(5566)
           │            ├─{Xvnc}(5567)
           │            ├─{Xvnc}(5568)

子Shell产生的途径

通过后台作业:&

[[email protected] Test]# cat jobs.sh
#!/bin/bash
parentShell="ParentShell"
echo "Parent Shell start and Level:"$BASH_SUBSHELL
# define subshell
{
 echo "SubShell start and Level:"$BASH_SUBSHELL
 subShell="SubShell"
 echo "SubShell value: ${subShell}"
 echo "parentShell value: ${parentShell}"
# sleep 5
 echo "SubShell end and Level: $BASH_SUBSHELL "
} & # running in backgroud
echo "Parent end and Level:$BASH_SUBSHELL"

if [ -z "${subShell}" ]
  then
    echo "subShell is not defined in ParentShell"
else
  echo "subShell is defined in ParentShel"
fi
[[email protected] Test]# bash jobs.sh
Parent Shell start and Level:0
Parent end and Level:0
subShell is not defined in ParentShell
SubShell start and Level:1
SubShell value: SubShell
parentShell value: ParentShell
SubShell end and Level: 1

根据运行结果,结论如下所示:

  • 在Shell中可以使用&产生子Shell
  • &产生的子Shell可以直接引用父Shell的变量,而子Shell产生的变量不能被父Shell引用
  • 在Shell中使用&可以实现多线程并发

通过管道:|

[[email protected] Test]# cat jobs.sh
#!/bin/bash
parentShell="ParentShell"
echo "Parent Shell start and Level:"$BASH_SUBSHELL
# define subshell
echo "" |  # 管道
{
 echo "SubShell start and Level:"$BASH_SUBSHELL
 subShell="SubShell"
 echo "SubShell value: ${subShell}"
 echo "parentShell value: ${parentShell}"
# sleep 5
 echo "SubShell end and Level: $BASH_SUBSHELL "
}
echo "Parent end and Level:$BASH_SUBSHELL"

if [ -z "${subShell}" ]
  then
    echo "subShell is not defined in ParentShell"
else
  echo "subShell is defined in ParentShel"
fi
[[email protected] Test]# bash jobs.sh
Parent Shell start and Level:0
SubShell start and Level:1
SubShell value: SubShell
parentShell value: ParentShell
SubShell end and Level: 1
Parent end and Level:0
subShell is not defined in ParentShell

根据运行结果,结论如下所示:

  • 在Shell中可以使用管道产生子Shell
  • 管道产生的子Shell可以直接引用父Shell的变量,而子Shell产生的变量不能被父Shell引用
  • 管道产生的Shell是顺序执行的,仅能在子Shell执行完成后才能返回父Shell中继续执行,这一点也是与&最大的区别。

通过()

[[email protected] Test]# cat jobs.sh
#!/bin/bash
parentShell="ParentShell"
echo "Parent Shell start and Level:"$BASH_SUBSHELL
# define subshell
(
 echo "SubShell start and Level:"$BASH_SUBSHELL
 subShell="SubShell"
 echo "SubShell value: ${subShell}"
 echo "parentShell value: ${parentShell}"
# sleep 5
 echo "SubShell end and Level: $BASH_SUBSHELL "
)
echo "Parent end and Level:$BASH_SUBSHELL"

if [ -z "${subShell}" ]
  then
    echo "subShell is not defined in ParentShell"
else
  echo "subShell is defined in ParentShel"
fi
[[email protected] Test]# bash jobs.sh
Parent Shell start and Level:0
SubShell start and Level:1
SubShell value: SubShell
parentShell value: ParentShell
SubShell end and Level: 1
Parent end and Level:0
subShell is not defined in ParentShell

根据运行结果,结论如下所示:

  • 在Shell中可以使用()产生子Shell
  • ()产生的子Shell可以直接引用父Shell的变量,而子Shell产生的变量不能被父Shell引用
  • ()产生的Shell是顺序执行的,仅能在子Shell执行完成后才能返回父Shell中继续执行,

看到这个结果,大家会不会觉得使用()跟使用管道一样的?

通过调用外部Shell

[[email protected] Test]# cat subShell.sh parentShell.sh  -n
       # SubShell
     1  #!/bin/bash
     2   echo "SubShell start and Level:"$BASH_SUBSHELL
     3   subShell="SubShell"
     4   echo "SubShell value: ${subShell}"
     5   echo "parentShell value: ${parentShell}"
     6   echo "parentExportShell value: ${parentExportShell}"
     7   if [ -z "${parentShell}"  ];then
     8      echo "parentShell value is : null"
     9   else
    10      echo "parentShell value is : "${parentShell}
    11   fi
    12
    13  # ParentShell
    14  #!/bin/bash
    15  parentShell="Parent"
    16  export parentExportShell="parentExportShell"
    17  echo "Parent Shell start and Level:"$BASH_SUBSHELL
    18  bash ./subShell.sh # invoke subshell
    19  sleep 3
    20  echo "Parent Shell end and Level:"$BASH_SUBSHELL
    21  if [ -z "${subShell}" ]
    22    then
    23     echo "subShell is not defined in ParentShell"
    24  else
    25     echo "subShell is defined in ParentShell"
    26  fi
[[email protected] Test]# bash parentShell.sh
Parent Shell start and Level:0
SubShell start and Level:0
SubShell value: SubShell
parentShell value:
parentExportShell value: parentExportShell
parentShell value is : null
Parent Shell end and Level:0
subShell is not defined in ParentShell

根据运行结果,结论如下所示:

  • 在Shell中可以通过外部Shell脚本产生子Shell
  • 在调用外部Shell时,父Shell定义的变量不能被子Shell继承,如果要继承父Shell的变量,必须使用export使其成为全局环境变量。
  • 调用外部Shell产生的Shell是顺序执行的,仅能在子Shell执行完成后才能返回父Shell中继续执行,

Shell脚本调用模式

? ? 通常在大型的项目中,都会将较大模块进行拆分为多个小模块进行代码编写调试等。因此在一个Shell脚本中也不可能包含所有模块,一般都采用在一个脚本中去调用当前用到的脚本,这种被称之为Shell嵌套。在一个脚本中嵌套脚本的方式主要有forkexecsource

fork模式调用脚本

? ? fork模式是最普通的脚本调用方式。在使用该方式调用脚本时,系统会创建一个子Shell去调用脚本。其调用方式如下所示:

/bin/bash /path/shellscript.sh # 未给脚本添加执行权限时
或
/path/shellscript.sh # 脚本拥有执行权限时

fork本质是复制进程。使用该方式时,fork会复制当前进程做为一个副本,而后将这些资源交给子进程。因此子进程会继承父进程的一些资源,如环境变量、变量等。而父进程却是完全独立的,子进程和父进程相当于面向对象中一个对象的两个实例。

exec模式调用脚本

? ? exec调用脚本时,不会开启一个新的子Shell来进行调用脚本,被调用的脚本和调用脚本在同一个Shell内执行。但需要注意的是使用exec调用新脚本后,在执行完新脚本的内容后,不再返回到调用脚本中执行后续未执行的内容,这也是与fork调用脚本的主要区别。其主要调用方式如下所示:

exec /path/shellscript.sh

exec的本质是加载另外一个程序来代替当前运行的进程。即在不创建新进程的情况下去加载一个新程序,而在进程执行完成后就直接退出exec所在的Shell环境。

source模式调用脚本

? ? source调用脚本时,也不会开启一个新的子Shell来执行被调用的脚本,同样也是在同一个Shell中执行,因此被调用脚本是可以继承调用脚本的变量、环境变量等。与exec调用方式的区别是,source在执行完被调用脚本的内容后,依然会返回调用脚本中,去执行调用脚本中未执行的内容。其主要调用方式如下所示:

source /path/shellscript.sh
或
. /path/shellscript.sh  # .和source是等价的

三种调用模式示例

? ? 示例代码如下所示:

[[email protected] Test]# cat -n subShell.sh parentShell.sh
     1  #!/bin/bash
     2   echo "SubShell start and Level:"$BASH_SUBSHELL
     3   echo "SubShell PID is:" $$
     4   subShell="SubShell"
     5   echo "SubShell value: ${subShell}"
     6   echo "parentShell value: ${parentShell}"
     7   echo "parentExportShell value: ${parentExportShell}"
     8   if [ -z "${parentShell}"  ];then
     9      echo "parentShell value is : null"
    10   else
    11      echo "parentShell value is : "${parentShell}
    12   fi
    13  #!/bin/bash
    14  # print usage
    15  function Usage() {
    16    echo "Usage:$0 {fork|exec|source}"
    17    exit 1
    18  }
    19  # print return variable
    20  function PrintPara() {
    21   if [ -z "${subShell}" ]
    22    then
    23     echo "subShell is not defined in ParentShell"
    24   else
    25     echo "subShell is defined in ParentShell "${subShell}
    26   fi
    27  }
    28  # invoke pattern
    29  function ParentFunction() {
    30    parentShell="Parent"
    31    export parentExportShell="parentExportShell"
    32    echo "Parent Shell start and Level:"$BASH_SUBSHELL
    33    echo "Parent PID is:"$$
    34    case "$1" in
    35      fork)
    36         echo "Using fork pattern"
    37         /bin/bash ./subShell.sh
    38         PrintPara ;;
    39      exec)
    40         echo "Using exec pattern"
    41         exec ./subShell.sh
    42         PrintPara ;;
    43      source)
    44         echo "Using source pattern"
    45         source ./subShell.sh
    46         PrintPara ;;
    47      *)
    48        echo "Input error ,usage is:" Usage
    49     esac
    50  }
    51  # check parameter number
    52  function CheckInputPara() {
    53    if [ $# -ne 1 ]
    54      then
    55        Usage
    56    fi
    57    ParentFunction $*
    58  }
    59  CheckInputPara $*

1、fork调用结果:

[[email protected] Test]# bash parentShell.sh fork
Parent Shell start and Level:0
Parent PID is:26413
Using fork pattern
SubShell start and Level:0
SubShell PID is: 26414
SubShell value: SubShell
parentShell value:
parentExportShell value: parentExportShell
parentShell value is : null
subShell is not defined in ParentShell

1、父Shell和子Shell的PID不一样,则可以说明产生了新的子进程

2、调用脚本中定义的全局变量可以传入到被调用脚本,而被调用的脚本中定义的变量是无法返回到调用脚本中的

2、exec调用结果:

[[email protected] Test]# chmod +x subShell.sh
[[email protected] Test]# bash parentShell.sh exec
Parent Shell start and Level:0
Parent PID is:25543
Using exec pattern
SubShell start and Level:0
SubShell PID is: 25543
SubShell value: SubShell
parentShell value:
parentExportShell value: parentExportShell
parentShell value is : null

1、父Shell和子Shell的PID一样,则可以说明未产生新的子进程

2、调用脚本中定义的全局变量可以传入到被调用脚本

3、最重要的一点就是在执行完被调用脚本后直接退出Shell了,而调用脚本未执行的内容并没有被执行

3、source调用结果:

[[email protected] Test]# bash parentShell.sh source
Parent Shell start and Level:0
Parent PID is:19955
Using source pattern
SubShell start and Level:0
SubShell PID is: 19955
SubShell value: SubShell
parentShell value: Parent
parentExportShell value: parentExportShell
parentShell value is : Parent
subShell is defined in ParentShell: SubShell

1、父Shell和子Shell的PID一样,则可以说明未产生新的子进程

2、调用脚本中定义的普通变量和全局变量可以传入到被调用脚本,反之亦然

3、最重要的一点就是在执行完被调用脚本后返回调用脚本中继续执行剩下的内容

三种调用模式使用场景

  • 1、fork模式使用场景

fork模式常用于常规嵌套脚本执行的场景中,仅仅是执行嵌套脚本中命令,调用脚本也不需要使用被调用脚本中的变量和函数等信息。

  • 2、exec模式使用场景

exec模式常用于调用脚本中末尾中。而这种模式在执行完被调用脚本中就直接退出,因此使用较少,可使用source代替exec

  • 3、source模式使用场景

source模式算是在Shell常用的一种脚本嵌套调用模式,常用于执行嵌套脚本启动一些服务程序等,其最大的优点就是嵌套脚本中定义的变量和函数等资源可以被调用脚本获取和使用。

本文同步在微信订阅号上发布,如各位小伙伴们喜欢我的文章,也可以关注我的微信订阅号:woaitest,或扫描下面的二维码添加关注:

原文地址:https://www.cnblogs.com/surpassme/p/10029758.html

时间: 2024-10-07 09:10:56

Shell编程-11-子Shell和Shell嵌套的相关文章

shell编程中的条件判断(shell 05)

shell编程中的条件判断条件if-thencase if-then单条件if commandthen commandsfi当command返回码为0时 条件成立 if.sh #! /bin/bash if date then echo "command exec" fi if date123 then echo "command exec1" fi echo "out if" [[email protected] sh]# ./if.sh 20

《跟老男孩学Linux运维之shell编程实战》-第二章 shell变量的核心基础

这篇文章主要讲解 shell变量的核心基础. 1.变量是什么? 变量是什么?可能有好多人不明白,简单地说,变量就是用一个固定的字符串(也可能是字符.数字等的组合)代替更多.更复杂的内容,该内容里可能还会包含变量.路径.字符串等其他的内容. 变量的赋值方式为:先写变量名称,紧接着是"="这个字符,最后是值,中间无任何空格(变量的内容一般要加双引号,以防止出错,特别是当值里的内容之间有空格时). 如何打印变量?通过echo命令加上$变量名 打印变量的值: 例如:定义变量和打印变量: [[e

《跟老男孩学Linux运维之shell编程实战》-第一章 shell脚本初步入门

本文是在学习<跟老男孩学Linux运维之shell编程实战>这本书时记录的知识点.看了这本书,我受益匪浅,当然这仅是我个人观点.下面我们言归正传,开始了解一下shell脚本吧! shell本身是一个命令解释器,它的作用是解释执行用户输入的命令及程序等. shell脚本语言的种类:sh.ksh.bash.csh.tcsh,Linux中主流的shell是bash,所以本文及后续shell脚本以bash为主. 那我们如何查看Linux系统中默认的shell? [[email protected] ~

小鸟初学Shell编程(一)认识Shell

开篇介绍 Linux里非常的有用的一个功能,这个功能就叫Shell脚本. Shell脚本在我日常开发工作里也占了非常重要的角色,项目中一些简单的工作我们都可以使用Shell脚本来完成,比如定时删除日志文件脚本.一键部署系统脚本等一些自动化的功能. 所以掌握好Shell脚本编程,处理一些自动化繁琐的事情会有事半功倍的效果.不管是Linux开发人员还是运维人员都有必要掌握Shell编程. 什么是Shell? Shell是命令解释器,用于解释用户对操作系统的操作. 简单的理解就是,Shell会把用户所

Linux Shell编程学习笔记一:shell简介

Q:什么是Shell,什么是shell脚本? A:Shell是一种用C语言编写的程序,它是用户与Linux操作系统沟通的桥梁.用户既可以输入命令执行,又可以利用 Shell脚本编程,完成更加复杂的操作.它是命令语言.命令解释程序及程序设计语言的统称.它的作用就是遵循一定的语法将输入的命令加以解释并传给系统. 命令解释程序:shell是一个命令语言解释器,它拥有自己内建的shell命令集,shell也能被系统中其他应用程序所调用.用户在提示符下输入的命令都由shell先解释然后传给Linux核心.

Linux shell编程(二):shell语法

shell 语法 shell 执行命令操作 当shell读取输入时,它将执行一系列操作.如果输入表示注释的开头,则shell将忽略注释符号('#')和该行的其余部分. shell 读取和执行命令时将经过下列操作: 从一个文件(Shell脚本)中读取它的输入,从作为参数的字符串到 "-c" 调用选项(调用Bash),或者从用户的终端. 将输入拆分为单词和操作符,遵守所描述的引用规则.这些标记由"元字符"分隔.此步骤执行别名扩展. 将令牌解析为简单的复合命令. 执行各种

shell编程中变量的运算 (shell 06)

主要包括以下3中 字符串操作数学运算浮点运算 一.字符串操作 字符串的连接 连接字2个字符串不需要任何连接符,挨着写即可 长度获取 expr length "hello" expr length "$str" 变量名必须放在双引号里,否者语法错误 查找字符串中字符的位置 expr index "$str" CHARS 第一个是从1 开始的,查找不到返回 0 ,返回匹配到的第一个字符的位置 [[email protected] ~]# echo $

【免费下载】全套最新 3.Shell编程 视频教程+教学资料+学习课件+源代码+软件开发工具

3.Shell编程视频教程 网盘地址: 链接:https://pan.baidu.com/s/1-M4wHTMaE8rxjroTTd-9qQ 提取码:qf52 加公众号 获取更多新教程 教程目录大纲 ./3.Shell编程 ├── 11 云计算-Linux系统管理-shell基础 │?? ├── 100 -云计算-Linux系统管理-shell基础-echo命令.avi │?? ├── 101 -云计算-Linux系统管理-shell基础-脚本执行方法.avi │?? ├── 102 -云计算-

老男孩新书Shell编程实战预定开始了-包邮包答疑

老男孩新书Shell编程实战预定开始了(扫图中二维码购买包邮定期包答疑),亲爱的伙伴们,让你们久等了.电脑用户猛戳 电脑购买链接 注意:包邮包3个月(2017春节后起算)Q群书籍内容答疑,等于白送书,还倒搭答疑服务. 如果觉得给力,请帮忙朋友圈.QQ群.空间转载! 活动优惠仅限本次活动,从其他渠道购买不享受此次的服务承诺. 定价 89元 基本信息 作者: 老男孩 丛书名: Linux/Unix技术丛书 出版社: 机械工业出版社 ISBN:9787111556077 上架时间:2017-1-6 出

Linux运维都要会哪些shell编程技能?

在充斥着各种的互联网+的数字时代,Linux运维也越来越趋于自动化方向发展,越来越多的运维工作者奔跑在了自动化运维的路上.目前Linux系统下最流行的运维自动化语言就是shell和Python了. 两者之间,shell几乎是IT企业必须使用的运维自动化编程语言,特别是在运维工作中的服务监控.业务快速部署.服务启动停止.数据备份及处理.日制分析等环节里,shell是不可缺的.当然Python也是一门更适合处理复杂的业务逻辑,以及开发复杂的运维软件工具,实现通过web访问等.Shell是一个命令解释