如何解决更正fd的shell提示
我正在用C创建自定义外壳,我想知道应该在哪个fd上编写提示。
mycoolshell $
查看其他经典shell,我发现dash
使用STDERR作为提示。 csh
和tcsh
使用STDOUT。对于bash
,zsh
和BSD sh
,我什么都找不到。我用过
% dash 2>file
echo qwe
echo qwe
% cat file
(dashprompt$)
(dashprompt$)
检查破折号的提示fd。与csh 1>file
的csh相同,但是我对其他的不幸。
是否为此使用标准或POSIX fd?可以使用STDIN吗?
解决方法
如果希望与Posix兼容,则需要将提示写到stderr
。 (请参见下面的PS1
环境变量的规范。)
不管严格的Posix兼容性如何,stdin
绝对是不正确的,因为它可能不允许写操作。 stdout
也不是一个好主意,因为它通常是行缓冲的。有些外壳程序(我相信包括zsh
)会将提示写到连接到当前终端(例如/dev/tty
)的文件描述符中,这可能是打开stderr
的样子,好像没有重定向一样,尽管它不一定是相同的文件描述符。但是使用/dev/tty
或同等功能是非标准的。
仅当外壳为 interactive 时才打印提示。根据Posix的说法,如果外壳是通过以下两种方式之一调用的,则它是交互式的:
如果存在
-i
选项,或者没有操作数,并且外壳程序的标准输入和标准错误附加到终端,则该外壳程序被认为是交互式的。 (sh
utility,Options)
很明显,如果您使用shell执行脚本,则不希望shell弹出提示。因此,您需要某种机制来判断该shell是交互使用还是作为脚本处理器使用。 Posix的要求似乎相当准确。 (请参阅isatty()
库函数以查看执行此测试的一种方法。)
这也说明了为什么stderr
重定向到文件时测试未能捕获提示。重定向stderr
会导致Shell非交互性,因此不会出现提示。为了正确进行测试,您需要使用-i
选项强制外壳交互。
Posix要求通过更改PS1
环境变量的值来修改提示。这是Posix所说的,包括要求将提示打印到stderr
的要求:(强调)
PS1
每次交互式外壳程序准备好读取命令时,此变量的值应进行参数扩展并写入标准错误。默认值应为“ $”。对于具有特定的其他实施定义特权的用户,默认值可能是另一个实施定义的值。外壳程序应替换字符“!”的每个实例。在PS1中,带有要键入的下一个命令的历史文件编号。转义“!”和另外一个 '!' (即“ !!”)应放置文字字符'!'在提示中。 (Shell command language,Shell Variables)
大多数外壳程序在PS1
中允许使用更丰富的替换集。但是,该值可以进行参数扩展这一事实允许进行广泛的自定义。这就意味着出现在PS1
变量值中的参数和命令引用(与通常的变量扩展不同)在每次打印提示时都会被扩展。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。