Linux终端提示符(prompt)不如期生效的原因分析与解决

2019-01-16 23:28:53王旭

前言

在开始本文的内容之前,我们先来简单介绍下, prompt是什么鬼?

顾名思义就是提示符的意思, 看起来和我们遥远, 但实际上只要是每个接触shell的童鞋, 都有看到, 那就是我们在输命令时前面的那串提示符.

例如:


当然, 这个样式是可以修改的, 这就涉及到我们的PS1和PS2了, 有经验或者以前有设置过的童鞋估计都不会陌生, 木有接触过的童鞋可以参考一下链接学习下: //www.jb51.net/article/72004.htm

问题

因为上面的并不是这次的主题, 所以简单跳过. 其实我们的prompt除了能够展示这些信息之外, 还能够展示些较为'动态'的东西.

我们先来看个需求(简单):

想要在每次执行命令时, 看到当前的时间,所以我们可以在~/.bash_profile里面加入下面的内容:

export PS1="$(date +"%Y-%m-%d %H:%M:%S")$ "

如代码所示, 应该能够在提示符上看到我们的时间, 现在试验下:

正如我们猜测的一样, 提示符那边已经变成了我们的时间了, 但是貌似出现了一个问题, 当我们在执行下一条命令时, 时间却没有改变, 还是40秒, 哪怕我们多执行几次还是这样.

难道PS1只能加载一次? 但是这个原因很快就被否决, 因为当我们在切换用户时, 提示符的$会改变成#, 而且也有其他的例子(下面会举出来)证明, PS1是每次都会执行的. 所以问题只能在于我们写法中.

考虑到是不是写死命令的缘故, 于是就把date命令放到函数中, 看能否成功:

修改~/.bash_profile文件如下:

echo_time(){ date +"%Y-%m-%d %H:%M:%S" } export PS1="$(echo_time)$ "

结果是失败的, 貌似这样的用法也是不行的! 查阅了很多文档, 直到看到一篇文章说, 两边必须是单引号, 才恍然大悟, 开始有点理解为什么我的效果会失败.

可能这里会有童鞋不清楚linux的单引号和双引号的区别, 简单来说就是:

双引号: 让大部分的符号(例如*), 失去意义,变为普通的字符. 单引号: 让所有的符号, 都失去意义, 变为普通的字符.

举个简单的例子, echo "$a" , 可以得出变量a的值, 但是echo '$a' , 就真的只能输出$a. 具体的细节, 可以查阅<shell十三问>.

原因分析

所以我这边失败的原因就是, 我这里需要用的是单引号, 而不是双引号, 因为如果用双引号, PS1在赋值时, 就已经获得命令/函数的值了, 所以每次打印PS1, 都已经是具体的值了, 而不是一个命令/函数. 就好像: