为什么看似zsh
明白要做什么some_variable=1 command
,却又不明白some-variable=1 command
?例如:$ fs_screen=1 mpv --fs-screen=${fs_screen} someFile.mp4
工作正常,但$ fs-screen=1 mpv --fs-screen=${fs-screen} someFile.mp4
结果是zsh: command not found: fs-screen=1
我深入研究了 zsh 文档,发现了我认为的问题所在。摘自https://zsh.sourceforge.io/Doc/Release/Shell-Grammar.html#Precommand-Modifiers:
A simple command may be preceded by a precommand modifier, which will alter how the command is interpreted. These modifiers are shell builtin commands with the exception of nocorrect which is a reserved word.
-
The command is executed with a ‘-’ prepended to its argv[0] string.
但是,我在任何地方都找不到有关此内容的更多信息。它的目的是什么,是否可以禁用它,以便我可以使用some-variable
而不是some_variable
?
-
在 zsh 变量名中无效并且永远不会有效,因为${var-foo}
是来自 Bourne shell 和 POSIX 的参数扩展运算符,扩展为foo
if$var
未设置$(( var-1 ))
来自 Korn shell 和 POSIX 的目的是进行减法算术运算。变量名中的有效字符是数字和下划线,并且不能以数字开头。
var=1
invar=1
or</dev/null var=1 cmd
被识别为赋值,因为它出现在命令参数之前,并且=
前面有一个未加引号的有效变量名。在
echo var=1
,\var=1
,var\=
,'v'ar=1
,a+b=c
,中a-b=c
,由于各种原因(在命令参数、变量名称之后或=
引用或部分引用、无效变量名称之后),这些不被视为赋值,因此它们被视为正常命令参数,并且要运行的命令由第一个参数得出在 中
fs-screen=1
, asfs-screen
不是有效的 shell 变量名,因此不会被识别为赋值,因此将其视为命令参数,并且由于它是第一个参数,因此将其视为命令的名称。因此,它将在 中的内置函数、函数、别名和可执行文件列表中查找$PATH
,如果没有找到,则会出现错误。您可以
fs-screen=1
使用以下方式定义函数:(《谍影重重》)或者:
(《科恩》)。
或者别名:
您还可以定义一个函数或一个名为 的可执行文件
var=1
,但要调用它,您需要引用=
变量名称或变量名称的任何部分(或在可执行文件中添加诸如env
或 之类的前缀command
)以避免将其视为赋值:在 zsh 中,赋值还可以是:
var[index]=value
分配给数组元素或字符串中的字符var[first,start]=value
对于范围或元素或字符相同1=foo
相同argv[1]=foo
:与其他类似 Bourne 的 shell 不同,位置参数可以被视为变量var=(1 2)
是一个数组或作为一个整体的关联数组赋值。新版本还有assoc=( [foo]=bar )
一个笨重的替代方案来assoc=( foo bar )
兼容 ksh93/bash。5.9.x 之后的下一个版本的 zsh 也将具有命名空间,因此您将能够执行此操作
a.x=foo
(但不是a.b.c=foo
;ksh93 已经具有命名空间以及复合变量和多维数组,并且还允许在数组索引内使用空格,因此您可以执行以下操作a[1 + 1][4].x[foo].bar=13
(即使 POSIX 要求运行该a[1
命令))。现在请注意,while
-
在变量名称中无效,在大多数类 Unix 系统中,所有字符甚至非字符(NUL 除外)都=
可以出现在环境变量名称中。也可能(尽管通常并不容易)有一个空名称的环境变量。但是那些包含除数字或下划线以外的字符,或以数字开头或位于 zsh 不会从环境中导入的列表中的内容,例如并且
IFS
无法USERNAME
映射到 shell 变量:(这是一个环境变量名称,不仅包含
-
换行符)。喜欢
perl
或python3
可以访问它的语言:但也不是
zsh
任何其他类似 Bourne 的 shell,因为它们访问环境变量的方式是通过映射到它们自己的变量。a_b
被映射到$a_b
标量(不是数组或关联)shell 变量并给出属性,export
因为它来自环境,但a-b
没有映射。zsh 仍然将其保留在环境中,因此它将传递给它执行的命令(例如printenv
上面的命令)。其他一些外壳喜欢mksh
将它们从环境中移除。¹ 您可能会争辩说,
${var-name}
如果不是上一点,则可以使用那里,就像您可以使用$(( $! - 1 ))
or$(( ${!} - 1 ))
作为$!
参数,否则也是算术运算符(与,!
相同,当然还有, ...),但这些不是变量。变量的要点是你可以给它们赋值,虽然你可以代替,但你不能代替,因为它们在算术表达式求值之前就被扩展了。$?
$*
$#
$1
$2
$(( $var ))
$(( var ))
$(( $var = 1+1 ))
$(( var = 1+1 ))
$var