用Linux经常可以看到终端,unix是一个支持多用户的操作系统,每个用户需要有一个输入输出的设备用来和操作系统系统打交道,这样的设备就叫终端,从功能上讲是字符型的输入输出设备,从语义上讲是给用户提供操作计算机的硬件接口。
每一个shell(广义上每一个进程)都有一个控制终端,这个shell的标准输入、输出、标准错误输出都是在这个控制终端上进行的。执行tty可以看到当前shell的控制终端,执行ps -ax可以看到进程的控制终端。
至于这个终端的初始化,当你用键盘在安装了linux的机器上本地登录时,使用的是控制台终端(/dev/ttyn)。
对于ssh,telnet等远程登录程序而言,当你ssh到某个sshd服务器上去时,这个sshd会打开一个伪终端主设备,然后fork出一个子进程,在子进程中打开一个从设备,这样,主进程和子进程之间就可以通过伪终端的主从设备进行交流,任何从主设备的输入都会输出到从设备上,为何要这样通信而不是用共享内存等方式呢?
因为中间通信的网络层需要对用户透明,即不能让使用者感觉到“指定从客户端发到了sshd,然后sshd去执行将结果返回。”,使用主从伪终端之后,当sshd收到指令时会将指令输入到主设备,然后主设备会把执行输出到从设备,这样就相当于指令输入到了从设备,而从设备是和某个shell连接的,从而这个指令或者毫无意义的字符串就被发往了远程的shell去解释。这个过程的结果就是用户感觉是自己的键盘和显示器直接插在了远程机器上一样。
这样的设计使得sshd可以不care指令的解释执行,通过模拟输入输出终端来实现。
下图来自apue chapter19
未解决问题:
使用ssh使用如果使用 here document 会得到
Pseudo-terminal will not be allocated because stdin is not a terminal.
的提示,原因待查,mark。