从shell眼中看世界–展开和引用

这篇博客的目的在于加深对于展开和引用的理解,阐释展开和引用之间联系。

‘ ‘ :强引用,其中的变量引用不会被替换为变量值,而保持原字符串
       ” “:弱引用,其中的变量引用会被替换为变量值
    如果之前对于’ ‘和” “的理解只停留在上述两句话的这个阶段,那么读完这篇博客相信你一定会有所收获。(本文会告诉你什么叫做一万句顶一句。)

    展开
        0.展开机制说明                  
        1.路径名展开                 *
        2.波浪线展开                 ~
        3.算术表达式展开           $(()) $[]
        4.花括号展开                 {}
        5.参数展开                    $
        6.命令替换                    $()   ` `

    0.展开机制说明
        echo是一个shell内部命令,作用是在标准输出中打印出它的文本参数。这里会使用echo命令来对所有展开进行说明,所以这一部分会非常容易理解。

        # echo this is a test
        this is a test


        echo将文本参数直接打印出来。

        # echo *
        aa addr.data anaconda-ks.cfg bin Checkos.sh hostch.sh hosts.bak hosts.BAK input ls-output.txt ls.txt Pics ping1.sh poem.txt shoplist.data test1.sh testuser.sh testz.sh tongku.sh


        如何工作:
        这里echo并没有直接打印,而是列出了当前工作目录下的所有文件(除了隐藏文件)及目录。这说明在echo命令执行前,shell把展开成了另外的东西。在对于通配符的学习中可以知道匹配任意长度的任意字符。当回车键被按下时,shell在命令被执行前自动展开当前工作目录下任何符合条件的文件名,所以echo命令所接收到的不是,而是*所展开成的结果。这就解释了echo * 执行结果。

        
    1.路径名展开
        在展开机制说明中提到的展开就是一种路径名展开。
        举例说明

        [root@localhost ~]# ls
        aa               Checkos.sh  input          ping1.sh       testuser.sh
        addr.data        hostch.sh   ls-output.txt  poem.txt       testz.sh
        anaconda-ks.cfg  hosts.bak   ls.txt         shoplist.data  tongku.sh
        bin              hosts.BAK   Pics           test1.sh

        /root目录中内容
        

        [root@localhost ~]# echo h*
        hostch.sh hosts.bak hosts.BAK


        h*展开为目录下以h开头的文件名
       

        [root@localhost ~]# echo *h
        Checkos.sh hostch.sh ping1.sh test1.sh testuser.sh testz.sh tongku.sh


        *h展开为目录下以h结尾的文件名

        [root@localhost ~]# echo [[:upper:]]*
        Checkos.sh Pics


        [[:upper:]]*展开为目录下以大写字母开头的文件名

        所谓展开就是shell对于命令行参数进行解释,改变其字面意义上的内容。(理解展开的关键)
        路径名展开把参数展开为符合匹配模式的内容。

    2.波浪线展开
        波浪线~是有特殊含义的。shell会对其进行特定解释。
        两种用法:
        ~后跟用户名,展开为指定用户的家目录。
        ~,不指定用户名,展开为当前用户的家目录。

        [root@localhost ~]# echo ~
        /root
        [root@localhost ~]# echo ~cold
        /home/cold


        
        波浪线展开:shell对~作特殊的解释说明。

    3.算术表达式展开
        shell允许算术表达式通过展开来执行。其格式$((expression))或者$[expression]
        操作符:+、-、*、/、%、**
        注意事项:
        算术表达式中空格并不重要,并且表达式可以嵌套。
        一对括号可以把子表达式括起来。由此可以简化嵌套。
        举例说明

        [root@localhost ~]# echo $((2+2))
        4
        [root@localhost ~]# echo $[2+2]
        4
        [root@localhost ~]# echo $(($((5**2))*3))
        75
        [root@localhost ~]# echo $(((5**2)*3))
        75


        算术表达式展开:执行算术表达式,并用结果替换之

    4.花括号展开
        相当有趣的一种展开。我将其理解为每个选项都选择一次。
        通过实例更容易说明其作用。

        [root@localhost ~]# echo Front-{A,B,C}-Back
        Front-A-Back Front-B-Back Front-C-Back


        选项用“,”隔开,ABC三个选项

        [root@localhost ~]# echo Number_{1..5}
        Number_1 Number_2 Number_3 Number_4 Number_5


        一系列整数

        [root@localhost ~]# echo {Z..A}
        Z Y X W V U T S R Q P O N M L K J I H G F E D C B A


        一系列倒序大写字母

        [root@localhost ~]# echo a{A{1,2},B{3,4}}b
        aA1b aA2b aB3b aB4b


        嵌套模式

        注意事项:
        {}中不能嵌入空白字符。下面是C前加入一个空白符后的执行结果。

        [root@localhost ~]# echo Front-{A,B, C}-Back
        Front-{A,B, C}-Back


        {}可以嵌套。

        这种展开最普遍的应用是,创建一系列的文件或目录列表。

        [root@localhost Pics]# mkdir {2015..2017}0{1..9} {2015..2017}{10..12}
        [root@localhost Pics]# ls
        201501  201505  201509  201601  201605  201609  201701  201705  201709
        201502  201506  201510  201602  201606  201610  201702  201706  201710
        201503  201507  201511  201603  201607  201611  201703  201707  201711
        201504  201508  201512  201604  201608  201612  201704  201708  201712


        看,相当的快捷方便。

    5.参数展开
        $变量名 将变量替换为变量值

        [root@localhost Pics]# echo $USER
        root


        USER系统定义的变量,变量值为当前用户名

        [root@localhost Pics]# x=10
        [root@localhost Pics]# echo $x
        10


        给x赋值为10,$x会被shell自动替换为10

    6.命令替换
        命令替换允许我们把一个命令的输出作为一个展开模式来使用。
        格式:$(command)   `command`

        [root@localhost ~]# which man
        /usr/bin/man
        [root@localhost ~]# ls -l `which man`
        -rwxr-xr-x. 1 root root 102736 Jun 10  2014 /usr/bin/man
        [root@localhost ~]# ls -l $(which man)
        -rwxr-xr-x. 1 root root 102736 Jun 10  2014 /usr/bin/man


        注意:命令的输出会被直接当成参数。如果命令输出不能直接作为下一条命令的参数,将不能正常执行。
        例如

        [root@localhost ~]# which cp
        alias cp='cp -i'
            /usr/bin/cp
        [root@localhost ~]# ls -l `which cp`
        ls: invalid option -- '''
        Try 'ls --help' for more information.


        which cp输出除文件路径还有其他内容,ls命令报错。

    引用
    shell有许多方式可以完成展开,现在是时候学习怎么来控制展开了。

    [root@localhost ~]# echo this is a    test
    this is a test


    shell从echo命令的参数列表中删除多余的空格。

    [root@localhost ~]# echo The total is $100.00
    The total is 00.00


    参数展开把$1 的值替换为一个空字符串,因为1是没有定义的变量。
    这里shell自作主张的对输入的字符串进行转换,有时这并不是我们所期望的。
    shell提供了引用机制,来有选择地禁止不需要的展开。
    

    1.双引号
        文本放在放在双引号之中,shell使用的特殊字符除$、\、`之外都会失去它们的特殊含义,被当做特殊字符对待。
        回想上面讲到的关于展开的内容,~波浪线展开,{}花括号展开,*等通配符的路径名展开,单词分割都被禁止。算术表达式展开,参数展开和命令替换仍然有效。
        
        注:什么是单词分割
        在默认情况下单词分割会在单词中寻找空格、制表符和换行符,并把它们看作单词之间的界定符。它们只作为分隔符使用。
        举例说明

        [root@localhost ~]# echo this is a    test
        this is a test


        这里多个空格只是作为分隔符使用,在上面这个例子中shell认为给出了四个参数,分别是this、is、a、test,将这四个参数传递给echo命令,因此执行结果中看不到多出的空格符的影响。
        [root@localhost ~]# echo “this is a    test”
        this is a    test
        在这个例子中this is a    test 文本被包含在””中,单词分割被禁止,内嵌的空格不再被当做界定符,而被作为参数的一部分。因此运行结果中输入的空格被保留了下来。

        这里还有一个关于单词分割的有趣例子,给出运行结果。

        [root@localhost ~]# echo $(cal)
        April 2017 Su Mo Tu We Th Fr Sa 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@localhost ~]# echo "$(cal)"
             April 2017     
        Su Mo Tu We Th Fr Sa
                           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


    2.单引号
        单引号禁止包含文本的所有展开。
        举例说明

        [root@localhost ~]# echo text ~/*.txt {a,b} $(echo cold) $((2+2)) $USER
        text /root/ls-output.txt /root/ls.txt /root/poem.txt a b cold 4 root


        所有展开都被允许。

        [root@localhost ~]# echo "text ~/*.txt {a,b} $(echo cold) $((2+2)) $USER"
        text ~/*.txt {a,b} cold 4 root


        只有命令替换,算术表达式展开,参数展开被允许。

        [root@localhost ~]# echo 'text ~/*.txt {a,b} $(echo cold) $((2+2)) $USER'
        text ~/*.txt {a,b} $(echo cold) $((2+2)) $USER


        所有展开都被禁止。

    3.转义字符
        在双引号中使用转义字符,可以有选择地阻止展开。
 

        [root@localhost ~]# echo "The balance for $USER is:$5.00"
        The balance for root is:.00   


        两个$都被当做参数展开。

        [root@localhost ~]# echo "The balance for $USER is:\$5.00"
        The balance for root is:$5.00


        阻止了$5.00的展开,\$被显示为$。

        在双引号中反斜杠失去它的特殊含义,被当做普通字符。
        

    总结
    《The Linux Command Line》中是这样说明这一章节的:
     随着我们继续学习 shell,你会发现使用展开和引用的频率逐渐多起来,所以能够很好的理解他们的工作方式很有意义。事实上,可以这样说,他们是学习 shell 的最重要的主题。 如果没有准确地理解展开模 式,shell 总是神秘和混乱的源泉,并且 shell 潜在的能力也 浪费掉了。

原创文章,作者:maru,如若转载,请注明出处:http://www.178linux.com/73499

(0)
marumaru
上一篇 2017-04-16 22:13
下一篇 2017-04-16 22:23

相关推荐

  • 0809linux基础小记(sed和vim用法)

    sed  [options] ‘scripts’ inputfile -n: 不输出模式空间内容的自动打印 -e: 多点编辑 -f:从指定文件中读取编辑脚本 -i:原处编辑 scripts: ’地址命令’ 地址定界: (1)  不给地址:对全文进行处理 (2)  单地址: #:  指定的行 例: sed '2…

    Linux干货 2016-08-11
  • 磁盘管理

    设备文件 I/O Ports: I/O 设备地址 一切皆文件: open(), read(), write(), close() 设备类型: 块设备:block ,存取单位“块”,磁盘 [root@localhost ~]# ll /dev/ brw-rw—-. 1 root disk 253, 0 Apr 29 03:53 dm-0 块设备 brw-r…

    2017-04-28
  • N25第五周总结:程序包管理

    程序包管理:rpm、yum、编译详解 rpm详解: 一:大纲        1、什么是rpm        2、为什么要使用rpm        3、rpm能实现什么功能 &nbs…

    Linux干货 2017-01-10
  • 磁盘配额、阵列与逻辑卷

    一、磁盘配额 在内核中执行以文件系统为单位启用对不同组或者用户的策略不同 根据块或者节点进行限制执行软限制(soft limit)硬限制(hard limit) 步骤一:获得文件系统支持临时有效:mount -o remount,usrquota,grpquota {mount_point|device}永久有效:写入配置文件 /etc/fstab 中检查文…

    Linux干货 2016-08-29
  • bash基础特性及用户与组管理

    1、列出当前系统上所有已经登录的用户的用户名,注意:同一个用户登录多次,则只显示一次即可。 2、取出最后登录到当前系统的用户的相关信息。 3、取出当前系统上被用户当做其默认shell的最多的那个shell。 4、将/etc/passwd中的第三个字段数值最大的后10个用户的信息全部改为大写后保存至/tmp/maxuser.txt文件中。 5、取出当前主机的I…

    2017-09-29
  • Linux用户与权限管理

    由于Linux是多用户、多任务系统,出于一些需要,当我们创建的文件希望对一部分用户开放,对一部分用户隐藏的时候应该怎么办呢?这样的话,就需要涉及到Linux系统中对用户以及对文件权限的管理。 一、相关文件 在Linux中,一切配置文件都是以文本文档的方式来保存的,同样用户的信息也保存在系统的一些文件中,其位置为/etc/passwd。此文件只有root用户具…

    Linux干货 2016-08-05

评论列表(1条)

  • renjin
    renjin 2017-04-21 09:40

    很明显,比上次写的博客都有一个大的提升,主要介绍了符号展开在linux shell中的用法,内容写的很详细,也很生动,排版还需要一个提升的过程,加油,加油