我刚刚开始使用Linux,我很好奇如何定义shell内置命令,如cd
。
此外,如果有人能解释他们如何实施和执行,我将不胜感激。
如果你想看看如何定义bash内置,那么你只需要看看Section 4 of The Bash Man Page。
但是,如果你想知道如何实现bash bultins,你需要查看the Bash source code,因为这些命令被编译成bash可执行文件。
查看命令是否为bash内置的一种快速简便的方法是使用help
命令。例如,help cd
将向您展示如何定义'cd'内置的bash。同样对于help echo
。
内置插件的实际集合因shell而异。有:
您可以使用type
命令查看该实用程序是否已内置,该命令受大多数shell支持(尽管其输出未标准化)。来自dash
的一个例子:
$ type ls
ls is /bin/ls
$ type cd
cd is a shell builtin
$ type exit
exit is a special shell builtin
Re cd
实用程序,理论上没有什么能阻止shell实现者将其实现为外部命令。 cd
不能直接更改shell的当前目录,但是,例如,cd
可以通过套接字将新目录传递给shell进程。但没有人这样做,因为没有意义。除了非常旧的炮弹(没有内置插件的概念),cd
使用一些脏系统黑客来完成它的工作。
如何在shell中实现cd
? here描述了基本算法。它还可以做一些工作来支持shell的额外功能。
Manjari,从ftp://ftp.gnu.org/gnu/bash/bash-2.05b.tar.gz检查bash shell的源代码你会发现shell内置命令的定义不是在单独的二进制可执行文件中,而是在shell二进制文件本身内部(内置名称shell清楚地暗示了这一点)。
每个Unix shell至少都有一些内置命令。这些内置命令是shell的一部分,并作为shell源代码的一部分实现。 shell认识到它被要求执行的命令是它的一个内置命令,并且它自己执行该操作,而不调用单独的可执行文件。不同的shell具有不同的内置,尽管在基本集中会有很多重叠。
有时,内置程序是出于性能原因而构建的。在这种情况下,$PATH
中通常还有该命令的一个版本(可能具有不同的功能集,不同的识别命令行参数集等),但shell决定将该命令作为内置实现,以便它可以从短暂的过程中拯救产生的工作,以完成它本身可以完成的工作。就像bash和printf一样,例如:
$ type printf
printf is a shell builtin
$ which printf
/usr/bin/printf
$ printf
printf: usage: printf [-v var] format [arguments]
$ /usr/bin/printf
/usr/bin/printf: missing operand
Try `/usr/bin/printf --help' for more information.
请注意,在上面的示例中,printf既是shell内置的(作为bash本身的一部分实现),也是外部命令(位于/ usr / bin / printf)。请注意,它们的行为也不同 - 当不带参数调用时,内置版本和命令版本会打印不同的错误消息。另请注意,-v var
选项(将此printf的结果存储到名为var
的shell变量中)只能作为shell的一部分完成 - 像/ usr / bin / printf这样的子进程无法访问执行它们的shell的变量。
这将我们带到故事的第二部分:一些命令是因为它们需要而构建的。一些命令,如chmod
,是系统调用的薄包装器。当你运行/bin/chmod 777 foo
时,shell forks,execs / bin / chmod(传递“777”和“foo”)作为参数,新的chmod进程运行C代码chmod("foo", 777);
然后将控制返回给shell。但这不适用于cd
命令。尽管cd
看起来和chmod
的情况相同,但它必须表现得不同:如果shell产生另一个进程来执行chdir
系统调用,它将仅为新生成的进程而不是shell更改目录。然后,当进程返回时,shell将保持原样位于同一目录中 - 因此cd
需要实现为内置shell。
壳牌内置 - 例如http://linux.about.com/library/cmd/blcmdl1_builtin.htm。 -
which cd
/usr/bin/which: no cd in (/usr/bin:/usr/local/bin......
不是内置的shell而是二进制的。
which ls
/bin/ls