分析家 发表于 2018-8-21 08:45:25

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

  在编写shell脚本的过程中,有的时候难免会用到一些变量值被定义为(*)的变量,但是当我们试图引用这个变量的时候bash有默认会把(*)替换成当前目录下的所有文件名的列表,如下:
# a=*  
# echo $a
  
anaconda-ks.cfg install.log install.log.syslog
  
# ls
  
anaconda-ks.cfginstall.loginstall.log.syslog
  这个时候我们可以考虑一个问题:这里的(*)是在哪一步被替换成当前目录下面的文件列表的呢:是在第一步,变量赋值的时候就被替换的呢还是说,在echo变量值的时候被替换的呢?
  事实是这样子的:
  1、当变量复制的时候,bash会直接将(*)赋值给变量a;
  2、但是在第二步引用变量的时候,bash默认会把(*)替换成当前目录下的所有文件的列表,大家可以这么实验一下:
# echo *  
anaconda-ks.cfg install.log install.log.syslog
  但是如何把变量a的值取出来呢,这个时候就把变量引用时引号的作用给体现出来了:
  当我们引用变量时,无引号、单引号、双引号的区别:()
# echo "$a"                #将引号里面的变量替换成相对应变量值  
*
  
# echo '$a'
  
$a                                        #将引号里面的字符统统不做转义,全部按字符串输出
  后面的问题接踵而至,当我想要在shell脚本中使用if语句判断某个变量的变量值是否为(*)的时候有报错了:
# [[ "$a" -eq * ]] && echo aa || echo bb  
-bash: [[: *: syntax error: operand expected (error token is "*")
  
bb
  于是想了一下,会不会是(*)没加双引号的原因呢:
# [[ "$a" -eq "*" ]] && echo aa || echo bb  
-bash: [[: *: syntax error: operand expected (error token is "*")
  
bb
  好吧,不行的,那把(-eq)换成(==)试下:
# [[ "$a" == "*" ]] && echo aa || echo bb  
aa
  
# [[ "$a" == * ]] && echo aa || echo bb
  
aa
  
# [[ $a == "*" ]] && echo aa || echo bb
  
aa
  
# [[ $a == * ]] && echo aa || echo bb
  
aa
  大家可能会奇怪,为什么最后一条都没加引号也是可以的呢,这里我们可以再做一个实验看下:
  我们将当前目录下的所有文件名的列表作为变量值赋给c
# echo $a  
anaconda-ks.cfg install.log install.log.syslog
  
# c="anaconda-ks.cfg install.log install.log.syslog"
  
# [[ "$c" == * ]] && echo aa || echo bb
  
aa
  
# [[ "$c" == "*" ]] && echo aa || echo bb
  
bb
  这个时候我们可以看到出,当两边都不加双引号的时候,(==)两边都是当前目录下的所有文件的列表作为变量值来进行比较,最后也是相等的;
  但是当一边加引号、一边不加的时候为什么也相等,暂时还解释不了,如果各位知道为什么,麻烦在下方的评论区告知一下,不胜感激。


页: [1]
查看完整版本: shell中变量被定义为星号(*)后无法引用的问题