首页 > 代码库 > shell中变量被定义为星号(*)后无法引用的问题

shell中变量被定义为星号(*)后无法引用的问题

    在编写shell脚本的过程中,有的时候难免会用到一些变量值被定义为(*)的变量,但是当我们试图引用这个变量的时候bash有默认会把(*)替换成当前目录下的所有文件名的列表,如下:

[root@vm_102 ~]# a=*
[root@vm_102 ~]# echo $a
anaconda-ks.cfg install.log install.log.syslog
[root@vm_102 ~]# ls 
anaconda-ks.cfg  install.log  install.log.syslog

    这个时候我们可以考虑一个问题:这里的(*)是在哪一步被替换成当前目录下面的文件列表的呢:是在第一步,变量赋值的时候就被替换的呢还是说,在echo变量值的时候被替换的呢?

    事实是这样子的:

    1、当变量复制的时候,bash会直接将(*)赋值给变量a;

    2、但是在第二步引用变量的时候,bash默认会把(*)替换成当前目录下的所有文件的列表,大家可以这么实验一下:

[root@vm_102 ~]# echo *
anaconda-ks.cfg install.log install.log.syslog

    但是如何把变量a的值取出来呢,这个时候就把变量引用时引号的作用给体现出来了:

当我们引用变量时,无引号、单引号、双引号的区别:()

[root@vm_102 ~]# echo "$a"                #将引号里面的变量替换成相对应变量值
*
[root@vm_102 ~]# echo ‘$a‘
$a                                        #将引号里面的字符统统不做转义,全部按字符串输出

    后面的问题接踵而至,当我想要在shell脚本中使用if语句判断某个变量的变量值是否为(*)的时候有报错了:

[root@vm_102 ~]# [[ "$a" -eq * ]] && echo aa || echo bb
-bash: [[: *: syntax error: operand expected (error token is "*")
bb

    于是想了一下,会不会是(*)没加双引号的原因呢:

[root@vm_102 ~]# [[ "$a" -eq "*" ]] && echo aa || echo bb
-bash: [[: *: syntax error: operand expected (error token is "*")
bb

    好吧,不行的,那把(-eq)换成(==)试下:

[root@vm_102 ~]# [[ "$a" == "*" ]] && echo aa || echo bb
aa
[root@vm_102 ~]# [[ "$a" == * ]] && echo aa || echo bb
aa
[root@vm_102 ~]# [[ $a == "*" ]] && echo aa || echo bb
aa
[root@vm_102 ~]# [[ $a == * ]] && echo aa || echo bb
aa

大家可能会奇怪,为什么最后一条都没加引号也是可以的呢,这里我们可以再做一个实验看下:

    我们将当前目录下的所有文件名的列表作为变量值赋给c

[root@vm_102 ~]# echo $a
anaconda-ks.cfg install.log install.log.syslog
[root@vm_102 ~]# c="anaconda-ks.cfg install.log install.log.syslog"
[root@vm_102 ~]# [[ "$c" == * ]] && echo aa || echo bb
aa
[root@vm_102 ~]# [[ "$c" == "*" ]] && echo aa || echo bb
bb

这个时候我们可以看到出,当两边都不加双引号的时候,(==)两边都是当前目录下的所有文件的列表作为变量值来进行比较,最后也是相等的;

    但是当一边加引号、一边不加的时候为什么也相等,暂时还解释不了,如果各位知道为什么,麻烦在下方的评论区告知一下,不胜感激。

本文出自 “红楼遗梦” 博客,请务必保留此出处http://leidongya.blog.51cto.com/7375845/1588056

shell中变量被定义为星号(*)后无法引用的问题