您的位置:首页 > 运维架构 > Shell

『忘了再学』Shell基础 — 10、Bash中的特殊符号(二)

2022-04-11 11:23 1991 查看

提示:本篇文章接上一篇文章,主要说说

()
小括号和
{}
大括号的区别与使用。

8、
()
小括号

()
:用于一串命令执行时,
()
中的命令会在子Shell中运行。(和下面大括号一起说明)

9、`

{}
:用于一串命令执行时,
{}
中的命令会在当前Shell中执行。也可以用于变量变形与替换。

(1)父Shell和子Shell

在介绍小括号和大括号的区别之前,我们先要解释一个概念,那就是父Shell和子Shell。

用户登录到Linux系统后,系统将启动一个用户Shell。在这个Shell中,可以使用Shell命令声明变量,也可以创建并运行Shell脚本程序。运行Shell脚本程序时,系统将创建一个子Shell。此时,系统中将有两个Shell,一个是登录时系统启动的Shell,另一个是系统为运行脚本程序创建的Shell。当一个脚本程序运行完毕,它的脚本Shell将终止,可以返回到执行该脚本之前的Shell。从这种意义上来 说,用户可以有许多Shell,每个Shell都是由某个Shell(称为父Shell)派生的。

在Linux系统中的默认Shell是

bash
,在
bash
中是可以调用新的
bash
的。在开启远程终端时候所启动的默认的交互Shell就是父Shell,只需要直接执行
bash
命令,就会创建一个新的Shell,这个Shell就是子Shell。

执行下面命令:

[root@localhost ~]# bash

我们就开启一个子Shell。

(2)区分父Shell子Shell

要区分是父Shell还是子Shell,需要使用前面所学的

ps
命令,查看进程命令来进行判断。

  • 执行
    ps -f
    查看系统进程。
  • 执行
    bash
    命令,开启一个新的Shell,并再次查看系统进程。 可以看到上图中,第二个进程的父
    id
    是第一个进程,所以第二个进程的
    bash
    是子Shell。
  • 退出子Shell,并再次查看系统进程。 可以看到子Shell进程结束,消失了。

(3)查看父子Shell的关系

我们可以通过

pstree
命令(查看进程树),来查看父子Shell的关系。

在Red Hat 6 中,所有的进程都是

init
进程的子进程。如下图:

我们可以看到在

init
进程下,开启了一个
sshd
的进程,这个进程就是远程登陆进程。我们执行过一次远程登陆
sshd
,和开启
bash
功能,在此
bash
下执行过一次
pstree
命令。

在当前Shell中,再执行一边

bash
命令,再次执行
pstree
命令(查看进程树)。 如下图:

我们可以从上图中看到,我在第一个

bash
下,又开启了一个新的
bash
,在新
bash
中执行了
pstree
命令。

所以说第一个

bash
是父(也就是父Shell),第二个
bash
是子(也就是子Shell)。

(4)父子Shell的关系拓展。

父Shell可以创建子Shell,在子Shell中还可以创建自己的子Shell。

它们的关系如下图所示:

下面来演示下这个关系图的Shell创建,和它们之前的关系层次。

下图中又创建了二个子Shell,总共三个子Shell。依次是子Shell, 孙Shell, 曾孙Shell。

下图使用

ps -f
命令,通过PPID列出谁是谁的父进程。

注意:生成子Shell的成本不低,而且速度还慢,创建嵌套的子Shell去处理命令进程性能更为严重。

通过输入

exit
命令能有条不絮的退出子Shell,例如上面的三个子Shell, 首先从曾孙Shell退出。 如下图所示:

注意:当没有了子Shell时,再输入

exit
命令,将退出控制台终端。

(5)小括号和大括号的区别

知道了父Shell和子Shell的关系,我们接着解释小括号和大括号的区别。

小括号和大括号的主要区别在于,在Shell程序执行的时候,小括号或者大括号中的内容是在父Shell执行还是在子Shell中执行。

下面我们总结一下小括号和大括号的主要区别:

  • ()
    中执行一串命令时,需要重新开一个子Shell进行执行。 在当前Shell中
    name=ss
    ,当执行到
    ()
    中命令的时候,会自动开启一个子Shell,在子Shell中
    name
    的变量赋予了
    mm
    ,当
    ()
    中命令运行完,子Shell进程就结束了,进程消失。里边的变量都不会保存,并自动返回到父Shell中,也就是回归到当前Shell,
    name
    的值还是原来Shell中赋予的
    ss
    值。 示例如下:
    #在父Shell中定义变量name的值是ss
    [root@localhost ~]# name=ss
    
    #如果用()括起来一串命令,这些命令都可以执行。
    #给name变量重新赋值,但是这个值只在子Shell中生效
    [root@localhost ~]#(name=mm;echo $name)
    mm
    
    #父Shell中name的值还是ss,而不是mm
    [root@localhost ~]# echo $name
    ss
  • { }
    中执行一串命令时,是在当前Shell执行。
    #用大括号来进行串命令的执行时
    #name变量的修改是直接在父Shell当中的
    #注意大括号的格式
    [root@localhost ~]# {  name=mm;echo $name; }
    mm
    
    #所以name变量的值已经被修改了
    [root@localhost ~]# echo $name
    mm
    就相当于直接在当前Shell执行
    {}
    中的语句,那么不写
    {}
    的结果是一样的。
  • ()
    {}
    都是把一串的命令放在括号里面,并且命令之间用
    号隔开。
  • ()
    最后一个命令可以不用分号结尾。
    [root@localhost ~]# ( name=lm; echo $name )
  • {}
    中最后一个命令要用分号结尾。
    [root@localhost ~]# { 空格 name=lm; echo $name; }
  • {}
    中的第一个命令和左括号之间必须要有一个空格。
    [root@localhost ~]# { 空格 name=lm; echo $name; }
  • ()
    里的各命令不必和括号有空格。
  • ()
    {}
    中,括号里面的某个命令的重定向只影响该命令,但括号外的重定向则影响到括号里的所有命令。

说明:小括号和大括号不太好理解,用的也不是太多,能看懂即可,工作中用不用在你自己。

10、
[]
中括号

[]
:用于变量的测试。(之后详解讲解)

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: