从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)
上一篇 2017-04-16 22:13
下一篇 2017-04-16 22:23

相关推荐

  • 何为正则表达式?

    何为正则表达式?   UNIX/Linux上有许多文本处理工具,其中最主要最重要要属grep、sed、和awk三种了,被称为文本处理三剑客。但是要完全认识他们的各种功能,则必须现在正则表达式及其元字符的使用上打好基础。 什么是正则表达式呢?正则表达式(regular expression,RE)是一种字符模式,用于在查找过程中匹配指定的字符。正则表…

    Linux干货 2016-08-16
  • linux上的LVM简明教程

    LVM是一个多才多艺的硬盘系统工具。在Linux上非常的好用,传统分区使用固定大小分区,重新调整大小十分麻烦。但是,LVM可以创建和管理“逻辑”卷,而不是直接使用物理硬盘。可以让管理员弹性的管理逻辑卷的扩大缩小,操作简单,而不损坏已存储的数据。可以随意将新的硬盘添加到LVM,以直接扩展已经存在的逻辑卷。 首先是实际的物理磁盘及其划分的分区和其上的物理卷(PV…

    Linux干货 2017-05-02
  • 进程管理和计划任务

    进程管理使用的工具以及命令,计划任务的创建和执行,以及工作中需要的注意事项

    2017-12-21
  • 马哥教育网络班22期+第6周课程练习

    week6: 请详细总结vim编辑器的使用并完成以下练习题 vim编辑器小结: 1. vim编辑器的模式: vim编辑器有很多模式,常用模式有:Normal,Insert,Command; 通常,打开一个文件,此时所处的就是normal模式;normal模式下可以浏览,修改文件内容; 在任何模式下,只要按ESC就可以返回到Normal模式; Ins…

    Linux干货 2016-09-26
  • 4th work

    1、复制/etc/skel目录为/home/tuser1,要求/home/tuser1及其内部文件的属组和其他用户均没有任何访问权限。 [root@localhost etc]# cp -r /etc/skel/ /home/tuser1 [root@localhost etc]# chmod go-rwx /home/tuser1/ 2、编辑/etc/gr…

    Linux干货 2017-10-09
  • 马哥教育网络班22期+第六周课程练习

    1、复制/etc/rc.d/rc.sysinit文件至/tmp目录,将/tmp/rc.sysinit文件中的以至少一个空白字符开头的行的行首加#; ]# cp -v  /etc/rc.d/rc.sysinit /tmp/    `/etc/rc.d/rc.sysinit' -> `/tmp/rc.sysinit&…

    Linux干货 2016-10-17

评论列表(1条)

  • renjin
    renjin 2017-04-21 09:40

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