📄 shell13问-全.doc
字号:
Return Value 的作用,是用来判断行程的退出状态(exit status),只有两种:
* 0 的话为"真"( true )
* 非 0 的话为"假"( false )
举个例子来说明好了:
假设当前目录内有一份 my.file 的文件,而 no.file 是不存在的:
CODE:[Copy to clipboard]$ touch my.file
$ ls my.file
$ echo $? # first echo
0
$ ls no.file
ls: no.file: No such file or directory
$ echo $? # second echo
1
$ echo $? # third echo
0
上例的第一个 echo 是关于 ls my.file 的 RV ,可得到 0 的值,因此为 true ﹔
第二个 echo 是关于 ls no.file 的 RV ,则得到非 0 的值,因此为 false ﹔
第三个 echo 是关于第二个 echo $? 的 RV ,为 0 的值,因此也为 true 。
请记住:每一个 command 在结束时都会送回 return value 的﹗不管你跑甚么样的命令...
然而,有一个命令却是"专门"用来测试某一条件而送出 return value 以供 true 或 false 的判断,
它就是 test 命令了﹗
若你用的是 bash ,请在 command line 下打 man test 或 man bash 来了解这个 test 的用法。
这是你可用作参考的最精确的文件了,要是听别人说的,仅作参考就好...
下面我只简单作一些辅助说明,其余的一律以 man 为准:
首先,test 的表示式我们称为 expression ,其命令格式有两种:
CODE:[Copy to clipboard]test expression
or:
[ expression ]
(请务必注意 [ ] 之间的空格键﹗)
用哪一种格式没所谓,都是一样的效果。(我个人比较喜欢后者...)
其次,bash 的 test 目前支持的测试对像只有三种:
* string:字符串,也就是纯文字。
* integer:整数( 0 或正整数,不含负数或小数点)。
* file:文件。
请初学者一定要搞清楚这三者的差异,因为 test 所用的 expression 是不一样的。
以 A=123 这个变量为例:
* [ "$A" = 123 ]:是字符串的测试,以测试 $A 是否为 1、2、3 这三个连续的"文字"。
* [ "$A" -eq 123 ]:是整数的测试,以测试 $A 是否等于"一百二十三"。
* [ -e "$A" ]:是关于文件的测试,以测试 123 这份"文件"是否存在。
第三,当 expression 测试为"真"时,test 就送回 0 (true) 的 return value ,否则送出非 0 (false)。
若在 expression 之前加上一个 " ! "(感叹号),则是当 expression 为"假时" 才送出 0 ,否则送出非 0 。
同时,test 也允许多重的覆合测试:
* expression1 -a expression2 :当两个 exrepssion 都为 true ,才送出 0 ,否则送出非 0 。
* expression1 -o expression2 :只需其中一个 exrepssion 为 true ,就送出 0 ,只有两者都为 false 才送出非 0 。
例如:
CODE:[Copy to clipboard][ -d "$file" -a -x "$file" ]
是表示当 $file 是一个目录、且同时具有 x 权限时,test 才会为 true 。
第四,在 command line 中使用 test 时,请别忘记命令行的"重组"特性,
也就是在碰到 meta 时会先处理 meta 再重新组建命令行。(这个特性我在第二及第四章都曾反复强调过)
比方说,若 test 碰到变量或命令替换时,若不能满足 expression 格式时,将会得到语法错误的结果。
举例来说好了:
关于 [ string1 = string2 ] 这个 test 格式,
在 = 号两边必须要有字符串,其中包括空(null)字符串(可用 soft quote 或 hard quote 取得)。
假如 $A 目前没有定义,或被定议为空字符串的话,那如下的写法将会失败:
CODE:[Copy to clipboard]$ unset A
$ [ $A = abc ]
[: =: unary operator expected
这是因为命令行碰到 $ 这个 meta 时,会替换 $A 的值,然后再重组命令行,那就变成了:
[ = abc ]
如此一来 = 号左边就没有字符串存在了,因此造成 test 的语法错误﹗
但是,下面这个写法则是成立的:
CODE:[Copy to clipboard]$ [ "$A" = abc ]
$ echo $?
1
这是因为在命令行重组后的结果为:
[ "" = abc ]
由于 = 左边我们用 soft quote 得到一个空字符串,而让 test 语法得以通过...
读者诸君请务必留意这些细节哦,因为稍一不慎,将会导至 test 的结果变了个样﹗
若您对 test 还不是很有经验的话,那在使用 test 时不妨先采用如下这一个"法则":
* 假如在 test 中碰到变量替换,用 soft quote 是最保险的﹗
若你对 quoting 不熟的话,请重新温习第四章的内容吧... ^_^
okay,关于更多的 test 用法,老话一句:请看 man page 吧﹗ ^_^
虽然洋洋洒洒讲了一大堆,或许你还在嘀咕.... 那... 那个 return value 有啥用啊?﹗
问得好﹗
告诉你:return value 的作用可大了﹗若你想让你的 shell 变"聪明"的话,就全靠它了:
* 有了 return value,我们可以让 shell 跟据不同的状态做不同的时情...
这时候,才让我来揭晓本章的答案吧~~~ ^_^
&& 与 || 都是用来"组建"多个 command line 用的:
* command1 && command2 :其意思是 command2 只有在 RV 为 0 (true) 的条件下执行。
* command1 || command2 :其意思是 command2 只有在 RV 为非 0 (false) 的条件下执行。
来,以例子来说好了:
CODE:[Copy to clipboard]$ A=123
$ [ -n "$A" ] && echo "yes! it's ture."
yes! it's ture.
$ unset A
$ [ -n "$A" ] && echo "yes! it's ture."
$ [ -n "$A" ] || echo "no, it's NOT ture."
no, it's NOT ture.
(注:[ -n string ] 是测试 string 长度大于 0 则为 true 。)
上例的第一个 && 命令行之所以会执行其右边的 echo 命令,是因为上一个 test 送回了 0 的 RV 值﹔
但第二次就不会执行,因为为 test 送回非 0 的结果...
同理,|| 右边的 echo 会被执行,却正是因为左边的 test 送回非 0 所引起的。
事实上,我们在同一命令行中,可用多个 && 或 || 来组建呢:
CODE:[Copy to clipboard]$ A=123
$ [ -n "$A" ] && echo "yes! it's ture." || echo "no, it's NOT ture."
yes! it's ture.
$ unset A
$ [ -n "$A" ] && echo "yes! it's ture." || echo "no, it's NOT ture."
no, it's NOT ture.
怎样,从这一刻开始,你是否觉得我们的 shell 是"很聪明"的呢? ^_^
好了,最后,布置一道习题给大家做做看、、、
下面的判断是:当 $A 被赋与值时,再看是否小于 100 ,否则送出 too big! :
CODE:[Copy to clipboard]$ A=123
$ [ -n "$A" ] && [ "$A" -lt 100 ] || echo 'too big!'
too big!
若我将 A 取消,照理说,应该不会送文字才对啊(因为第一个条件就不成立了)...
CODE:[Copy to clipboard]$ unset A
$ [ -n "$A" ] && [ "$A" -lt 100 ] || echo 'too big!'
too big!
为何上面的结果也可得到呢?
又,如何解决之呢?
(提示:修改方法很多,其中一种方法可利用第七章介绍过的 command group ...)
快﹗告我我答案﹗其余免谈....
11) >; 与 < 差在哪?
这次的题目之前我在 CU 的 shell 版已说明过了:
http://bbs.chinaunix.net/forum/24/20031030/191375.html
这次我就不重写了,将贴子的内容"抄"下来就是了...
--------------
11.1
谈到 I/O redirection ,不妨先让我们认识一下 File Descriptor (FD) 。
程序的运算,在大部份情况下都是进行数据(data)的处理,
这些数据从哪读进?又,送出到哪里呢?
这就是 file descriptor (FD) 的功用了。
在 shell 程序中,最常使用的 FD 大概有三个,分别为:
0: Standard Input (STDIN)
1: Standard Output (STDOUT)
2: Standard Error Output (STDERR)
在标准情况下,这些 FD 分别跟如下设备(device)关联:
stdin(0): keyboard
stdout(1): monitor
stderr(2): monitor
我们可以用如下下命令测试一下:
CODE:[Copy to clipboard]$ mail -s test root
this is a test mail.
please skip.
^d (同时按 crtl 跟 d 键)
很明显,mail 程序所读进的数据,就是从 stdin 也就是 keyboard 读进的。
不过,不见得每个程序的 stdin 都跟 mail 一样从 keyboard 读进,
因为程序作者可以从档案参数读进 stdin ,如:
CODE:[Copy to clipboard]$ cat /etc/passwd
但,要是 cat 之后没有档案参数则又如何呢?
哦,请您自己玩玩看啰.... ^_^
CODE:[Copy to clipboard]$ cat
(请留意数据输出到哪里去了,最后别忘了按 ^d 离开...)
至于 stdout 与 stderr ,嗯... 等我有空再续吧... ^_^
还是,有哪位前辈要来玩接龙呢?
--------------
11.2
沿文再续,书接上一回... ^_^
相信,经过上一个练习后,你对 stdin 与 stdout 应该不难理解吧?
然后,让我们继续看 stderr 好了。
事实上,stderr 没甚么难理解的:说穿了就是"错误信息"要往哪边送而已...
比方说,若读进的档案参数是不存在的,那我们在 monitor 上就看到了:
CODE:[Copy to clipboard]$ ls no.such.file
ls: no.such.file: No such file or directory
若,一个命令同时产生 stdout 与 stderr 呢?
那还不简单,都送到 monitor 来就好了:
CODE:[Copy to clipboard]$ touch my.file
$ ls my.file no.such.file
ls: no.such.file: No such file or directory
my.file
okay,至此,关于 FD 及其名称、还有相关联的设备,相信你已经没问题了吧?
那好,接下来让我们看看如何改变这些 FD 的预设数据信道,
我们可用 < 来改变读进的数据信道(stdin),使之从指定的档案读进。
我们可用 >; 来改变送出的数据信道(stdout, stderr),使之输出到指定的档案。
比方说:
CODE:[Copy to clipboard]$ cat < my.file
就是从 my.file 读进数据
CODE:[Copy to clipboard]$ mail -s test root < /etc/passwd
则是从 /etc/passwd 读进...
这样一来,stdin 将不再是从 keyboard 读进,而是从档案读进了...
严格来说,< 符号之前需要指定一个 FD 的(之间不能有空白),
但因为 0 是 < 的默认值,因此 < 与 0< 是一样的﹗
okay,这个好理解吧?
那,要是用两个 << 又是啥呢?
这是所谓的 HERE Document ,它可以让我们输入一段文本,直到读到 << 后指定的字符串。
比方说:
CODE:[Copy to clipboard]$ cat <<FINISH
first line here
second line there
third line nowhere
FINISH
这样的话,cat 会读进 3 行句子,而无需从 keyboard 读进数据且要等 ^d 结束输入。
至于 >; 又如何呢?
且听下回分解....
--------------
11.3
okay,又到讲古时间~~~
当你搞懂了 0< 原来就是改变 stdin 的数据输入信道之后,相信要理解如下两个 redirection 就不难了:
* 1>;
* 2>;
前者是改变 stdout 的数据输出信道,后者是改变 stderr 的数据输出信道。
两者都是将原本要送出到 monitor 的数据转向输出到指定档案去。
由于 1 是 >; 的默认值,因此,1>; 与 >; 是相同的,都是改 stdout 。
用上次的 ls 例子来说明一下好了:
CODE:[Copy to clipboard]$ ls my.file no.such.file 1>;file.out
ls: no.such.file: No such file or directory
这样 monitor 就只剩下 stderr 而已。因为 stdout 给写进 file.out 去了。
CODE:[Copy to clipboard]$ ls my.file no.such.file 2>;file.err
my.file
这样 monitor 就只剩下 stdout ,因为 stderr 写进了 file.err 。
CODE:[Copy to clipboard]$ ls my.file no.such.file 1>;file
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -