touch foo=3 # Create a file with name "foo=3" + touch foo=3 declare foo=? + declare 'foo=?' alias foo=* + alias 'foo=*'
我得不到我的期望,因为foo =?并且foo = *未扩展为文件名“foo = 3”:
declare -p | grep 'foo=' # => foo='?' alias | grep 'foo=' # => alias foo='*'
但是如果我运行另一个类似cd的内置函数或接受赋值作为自己编写的参数的函数,如show_rhs(){echo“${1%= *} =’${1#* =}’”;我得到了我期望的东西(foo =?和foo = *被扩展).
cd foo=? # => foo=3: Not a directory show_rhs() foo=* # => foo='3'
我在这里看到的唯一区别是declare和alias是内置函数并接受赋值作为参数.根据-x选项的输出,似乎添加了一对引用以在文件名扩展之前包含赋值.
但是如果文件名扩展在命令执行之前运行而不管命令是什么,那么传入declare和alias的参数应该是foo = 3而不是foo =?和foo = *由于存在文件“foo = 3”.
那么Bash在文件名扩展之前根据命令做一些特殊的事情(可能引用通配符?)到“a = b”类似的参数?
(我的环境:CentOS 5.8 64bit,GNU Bash 3.2.25)
解决方法
特别是,接受这些参数(别名,声明,导出,本地,只读和排版)的命令中的赋值参数不受路径名扩展或字拆分的限制. (这是通过抑制扩展而在内部完成的,而不是通过引用元字符来完成,尽管不容易看到实现细节如何变得可见.)
即使在Posix模式下启动bash或sh也是如此.
请注意,抑制路径名扩展仅适用于看起来像赋值的参数.从问题中扩展示例:
touch foo=3 # Create a file with name "foo=3" + touch foo=3 declare foo=? + declare 'foo=?' bar="foo=?" # Put the declare argument in a variable + bar='foo=?' declare $bar + declare foo=3
正如预期的那样,破折号路径名会扩展并将字段拆分为别名和导出的参数,与Posix规范一致.所以,显然,zsh.
除了在Posix模式下,bash还会向参数的右侧扩展 – 看起来像赋值.在Posix模式下,它将此限制为上面列出的内置函数的赋值参数,尽管Posix仅在命令字之前的变量赋值中指定=之后的波浪扩展.这就是破折号所做的,但是zsh将其扩展为“排版族的命令”(记录在zsh manual中).