Shell
参考教程MIT公开课——The Missing Semester of Your CS Education
What is the shell?
why we need shell?
These are great for 80% of use-cases(use GUI), but they are often fundamentally restricted in what they allow you to do — you cannot press a button that isn’t there or give a voice command that hasn’t been programmed.To take full advantage of the tools your computer provides, we have to go old-school and drop down to a textual interface: The Shell.
总而言之就是,图形化界面给我们提供了极大的便利,但由于将功能封装在一个个特定功能的按键等交互上,使得其灵活性大打折扣,而更加古老的命令行工具很大程度上能够解决这一问题。话说这结论我好像在哪听过,像是最近在读的Orange's:一个操作系统的实现
里也有相似论点。
Using the shell
在终端的命令提示符中输入想要执行的指令,这些命令会被shell
所解析,以下是两个较为简单的命令
frechen026@frechen026-virtual-machine:~$ date |
传入的参数以空格分隔,一个多单词的参数,我们可以使用引号或者转义字符处理
shell
借助环境变量$PATH
知道某个程序存放的位置
shell
说白了也是一种程序设计语言,提示符中甚至可以有条件、循环等等语句
可以通过echo $PATH
查看有哪些环境变量,同时使用which
指令可以知晓某程序所在路径,可以发现将echo
替换成/usr/bin/echo
效果相同。
frechen026@frechen026-virtual-machine:~$ echo $PATH |
path
用以描述计算机中文件所在位置,Linux
或MacOS
起始路径都为根目录,位于整个文件系统的最顶层,Windows
下每一个磁盘分区下都有一个根目录。Linux
下可使用pwd
指令打印出当前所在目录路径。
相对路径与绝对路径,Linux
课程笔记中已有备注,不加以赘述。
.
和..
是较为特殊两个目录,其实也都知道了,.
表示当前目录,..
表示当前目录的父目录。
cd
命令中两个较为特殊的符号~
和-
,~
等价于/home/username
,所以回到用户目录可以使用以下命令
frechen026@frechen026-virtual-machine:/$ cd ~ |
(其实直接cd
不加任何参数也行)
frechen026@frechen026-virtual-machine:/$ cd |
不过下列情形就有优势了
frechen026@frechen026-virtual-machine:~$ cd ~/Desktop/ |
-
可以切换至上一次cd
的目录,用于在两个目录之间跳转是极好的。
frechen026@frechen026-virtual-machine:~/Desktop$ cd - |
还有很多命令的讲解如ls
man
move
…都与Linux
相重复了,也就不加以赘述。
一个好用的快捷键Ctrl + l
清屏操作,我居然还只会傻乎乎用clear
指令,改天对快捷键也加以整理吧,无疑使用好快捷键可以大大提高效率。
在程序间创建连接
重定向
在 shell 中,程序有两个主要的“流”:它们的输入流和输出流。 当程序尝试读取信息时,它们会从输入流中进行读取,当程序打印信息时,它们会将信息输出到输出流中。 通常,一个程序的输入输出流都是您的终端。也就是,您的键盘作为输入,显示器作为输出。 但是,我们也可以重定向这些流
最简单的重定向是 < file
和 > file
。这两个命令可以将程序的输入输出流分别重定向到文件:
frechen026@frechen026-virtual-machine:~/Desktop$ echo hello > hello.txt |
上述指令实现了将本应该输出到终端的output stream
重定向到了hello.txt
中
frechen026@frechen026-virtual-machine:~/Desktop$ cat < hello.txt |
上述指令实现了将hello.txt
内容作为输入重定向到cat
指令中,效果等同于cat hello.txt
frechen026@frechen026-virtual-machine:~/Desktop$ cat < hello.txt > world.txt |
上述指令实现了将hello.txt
内容作为input stream
重定向给cat
指令作为输入,又将cat
指令的输出重定向给world.txt
作为输入。事实上cat
指令并不知道重定向这回事,他只管接受属于他的input stream
并输出它的output stream
,重定向这回事是shell
该干的事。
使用>>
来向一个文件追加内容,注意,不单单是写入,而是追加:
frechen026@frechen026-virtual-machine:~/Desktop$ cat < hello.txt >> world.txt |
pipe
|
分隔两个程序,将左侧程序的输出作为右侧程序的输入,看jyy第一节课就用到这个玩意的,当时也没去多想,看似高深实际了解了好像也就那么回事。
frechen026@frechen026-virtual-machine:~/Desktop$ ls -l |
照着教程敲了tail
,猜了一下对应应该有head
指令吧,还真有。嘿嘿,合理的命名确实易于理解和记忆。上述指令将ls -l
所输出的文件作为输入传递给了后续程序。
ls
指令并不认识tail
和head
,反之亦是如此,并非刻意兼容,他们所知道的只有从输入读数据,写到输出中,连接他们的是pipe
frechen026@frechen026-virtual-machine:~/Desktop$ ls -l |
上述shell
实现管道连接的同时将输出重定向到hello.txt
中
确实pipe
是个十分神奇的东西,将命令联合在一起,对于文本数据甚至图片、视频都可以处理吗?
root用户
对于大多数的类 Unix 系统,有一类用户是非常特殊的,那就是:根用户(root user)。 您应该已经注意到了,在上面的输出结果中,根用户几乎不受任何限制,他可以创建、读取、更新和删除系统中的任何文件。
通常在我们并不会以根用户的身份直接登录系统,因为这样可能会因为某些错误的操作而破坏系统。 取而代之的是我们会在需要的时候使用 sudo
命令。
Linux
中一切皆文件这个观点我们早就知道了,因此我们也可以对内核、外设那些相关文件进行直接的读写操作,这确实是件amazing的事情,尽管可能无法承受随意更改带来的后果。
有一件事情是您必须作为根用户才能做的,那就是向 sysfs
文件写入内容。系统被挂载在 /sys
下,sysfs
文件则暴露了一些内核(kernel)参数。 因此,您不需要借助任何专用的工具,就可以轻松地在运行期间配置系统内核。注意 Windows 和 macOS 没有这个文件?
这里我就纳闷了,Windows如今也支持Linux
,如我就在使用的wsl2
,利用它我可以不用虚拟机,不用双系统,就能在Windows上拥有Linux
,wsl2
的Ubuntu20.04
我也看了,也是存在该文件的,虚拟机下也存在该文件,不过其中包含的内容却有些许差异,就比如两者都没有例子中所提及的brightness
,所以下面所尝试的例子我本人并没有实际去实验。
sudo find -L /sys/class/backlight -maxdepth 2 -name '*brightness*' |
This error may come as a surprise.我们明明已经使用了sudo
去执行相关的操作,却仍然得到了有关于权限的报错,sad!But why?
先前对于重定向以及管道的说明就以提及,输入输出的重定向是程序所不知道的,这些都是由shell
所设计好的。echo
等程序并不知道 |
的存在,它们只知道从自己的输入输出流中进行读写。
额,这里中文翻译的感觉不是太过于确切。以下是中英对照:In the case above, the shell (which is authenticated just as your user) tries to open the brightness file for writing, before setting that as sudo echo’s output, but is prevented from doing so since the shell does not run as root.
对于上面这种情况, shell (权限为您的当前用户) 在设置 sudo echo 前尝试打开 brightness 文件并写入,但是系统拒绝了 shell 的操作因为此时 shell 不是根用户。
shell (权限为您的当前用户) 在设置 sudo echo 前尝试打开 brightness 文件并写入
翻译的过于令人误解,应当是再将其设置为sudo echo
的输出前,shell
尝试去打开并写入该文件。
echo 3 | sudo tee brightness |
我们可以尝试上述写法,由tee
获得了root
权限,由它去写brightness
,在整条指令前面加上#
,表示以root
权限执行整条指令,也可解决该问题。这样您就可以在 /sys
中愉快地玩耍了,例如修改系统中各种LED的状态(路径可能会有所不同)
这里其实还是很迷糊的,因为该节课并未对shell
是什么加以很明确的说明,虽然它的第一部分是what is the shell
,但遗憾的是不管是课件还是讲解都未对shell
加以说明,只知道他是古老的文本接口,它可以在终端中运行,可它究竟是什么?与终端有何区别却是缺失的。
课后习题:
1.To make sure you’re running an appropriate shell, you can try the command echo $SHELL.
frechen026@pine64:~$ echo $SHELL |
2.Create a new directory called missing under /tmp.
frechen026@pine64:/tmp$ ls |
3.Look up the touch program. The man program is your friend.
frechen026@pine64:/$ man touch |
4.Use touch to create a new file called semester in missing.
frechen026@pine64:/tmp$ cd missing |
5.Write the following into that file, one line at a time:
#!/bin/sh |
frechen026@pine64:/tmp/missing$ echo '#!/bin/sh' >> semester |
使用追加重定向解决,不过直接使用文本编辑器显然更简单,例如vim
,这也不算是跳脱命令行。
不顾提示,使用双引号进行尝试
frechen026@pine64:/tmp/missing$ echo "#!/bin/sh" >> semester |
得到如下报错,果然! has a special meaning even within double-quoted (") strings.
6.Try to execute the file, i.e. type the path to the script (./semester) into your shell and press enter. Understand why it doesn’t work by consulting the output of ls (hint: look at the permission bits of the file)
frechen026@pine64:/tmp/missing$ ./semester |
可以看到该文件并无执行权限,只有读写权限,所以无法执行。
frechen026@pine64:/tmp/missing$ vim hello.c |
快速编写一个可执行文件a.out
,权限可以看到是可执行的。
7.Run the command by explicitly starting the sh interpreter, and giving it the file semester as the first argument, i.e. sh semester. Why does this work, while ./semester didn’t?
frechen026@pine64:/tmp/missing$ cat semester |
哈?居然真的执行了相关的内容,这莫非就是shell
脚本的雏形?想来确实是执行了semester
文件中的语句,amazing!!!
8.Look up the chmod program (e.g. use man chmod).
frechen026@pine64:/tmp/missing$ man chmod |
根据名字判断chmod - change file mode bits
,应该可以改变用户对文件的权限。
9.Use chmod to make it possible to run the command ./semester rather than having to type sh semester. How does your shell know that the file is supposed to be interpreted using sh? See this page on the shebang line for more information.
frechen026@pine64:/tmp/missing$ chmod u+x semester |
为semester
加上可执行权限。有点意外,一个普通文件居然还能够执行,超出我对可执行文件的理解了。不过应该是shell
去执行了其中的语句,跟sh
同理。
10.Use | and > to write the “last modified” date output by semester into a file called last-modified.txt in your home directory.
frechen026@pine64:/tmp/missing$ ./semester | tail -n2 > last-modified.txt |
tail -n1
仅仅只是个换行,额,n2
看来才是符合要求的结果。
11.Write a command that reads out your laptop battery’s power level or your desktop machine’s CPU temperature from /sys. Note: if you’re a macOS user, your OS doesn’t have sysfs, so you can skip this exercise.
CPU
温度相关文件在/sys/class/thermal/
中
frechen026@pine64:/sys/class/thermal/thermal_zone0$ cat temp |
我去,燃爆了,居然温度高达83.98℃,好在我一阵风力降温
frechen026@pine64:/sys/class/thermal/thermal_zone0$ cat temp |
温度降到了54℃,不过小风扇没电了,温度也是又回到了80,难怪之前摸都烫手……
frechen026@pine64:/sys/class/power_supply/axp20x-battery$ cat capacity |
查看电池,额,插电使用的卡片电脑,我也不晓得这电量该怎么算,不过capacity
应该是容量吧,100应该是满电的。
至此学习完了第一节课的内容。
补充概念
回到上述提及过的问题,shell
究竟是什么?其与terminal
的联系与区别在哪?
什么是shell
Shell
是一个用 C 语言
编写的程序,它是用户使用 Linux
的桥梁。Shell
既是一种命令语言,又是一种程序设计语言。
Shell
是指一种应用程序,这个应用程序提供了一个界面,用户通过这个界面访问操作系统内核的服务。
我不清楚将shell
称之为一种语言是否合适,shell
确实有属于自己的shell
语言,但是否可以说shell∈programming language
我是不太赞同的,更像是C
和gcc
的关系。
找到较为权威的文档Bash Reference Manual,其中对于what is a shell
做了如下解释:
At its base, a shell is simply a macro processor that executes commands. |
显然,这里也确实将shell
称为a programming language
,那也无话可说,shell
就姑且定义为一个具有语言特性的解释器。
Ken Thompson
的 sh
是第一种 Unix Shell
,Windows Explorer
是一个典型的图形界面 Shell
。
Linux 的 Shell 种类众多,常见的有:
- Bourne Shell(/usr/bin/sh或/bin/sh)
- Bourne Again Shell(/bin/bash)
- C Shell(/usr/bin/csh)
- K Shell(/usr/bin/ksh)
- Shell for Root(/sbin/sh)
- ……
Bourne Again Shell
,由于易用和免费,Bash
在日常工作中被广泛使用。同时,Bash
也是大多数Linux
系统默认的 Shell
。
terminal
与shell
有何区别?
什么是terminal
终端是与计算机系统相连的一种输入、输出设备,通常离计算机较远。
早期的终端是一种叫做 电传打字机 - Teletype
的东西。虽然现在电传打字机已经不再适用,但它也是计算机发展的一个重要象征。所以 Teletype
的缩写 TTY
沿用至今,作为各种类型的终端设备的统称。难怪oranges
一书中对于terminal
使用的是tty
进行描述,但是还纳闷是什么单词的缩写,原来如此。
实际上,前面我们说的终端指的都是物理的终端设备,而我们在 Linux
上常说的终端其实是 终端模拟器 - Terminal Emulator
(一种 模拟终端的程序
),也称为 —— 虚拟终端
,但平时一般直接称为终端。
可以将 Linux
终端看作一个使用软件来模拟的输入、输出设备,其作用就是提供一个命令的输入、输出环境。Linux
终端是基于物理终端之上的。
shell
- 在计算机科学中,Shell 俗称 “壳” (区别于 “核”—— Linux 内核)
- 简单来说,Shell 就是接收用户输入的命令,然后提交给 Linux 内核处理的一个壳程序
shell
壳的名称确实及其贴切,相对于系统内核而言,Shell
是包裹在操作系统 外层 的一道程序,负责外界与 Linux “内核” 的交互,但它隐藏了操作系统底层的具体细节,就像是 Linux
内核的一个 “外壳”,所以 Shell
(壳)的名称也由此而来。
总结
由上述对于两者的定义不难看出,终端起到模拟输入输出的作用,提供人机交互的接口,将获得的输入转交给shell
,而shell
对获得的输入进行处理,交给Linux
内核加以执行。