专注各种脚本编程
Baidu
加入收藏夹
本站内容有下面分类知识,欢迎您的到来^_^
shell相关:指令篇 基础篇 脚本欣赏 编程实例 shell问问 shell视频教程 技巧篇 水平测试 E文资料 vi编辑器 高级Bash脚本编程指南
其他:mysql perl c语言 oracle
当前位置:| 主页>shell问问>

十一问:>与<差别在哪?(下篇)

百度收藏 QQ搜藏

--------------
11.4
    okay,这次不讲 I/O Redirction ,讲佛吧...(有没搞错?﹗网中人是否头壳烧坏了?...) 嘻~~~ ^_^
    学佛的最高境界,就是"四大皆空"。至于是空哪四大块?我也不知,因为我还没到那境界... 但这个"空"字,却非常值得我们返复把玩的:
   --- 色即是空、空即是色﹗
   好了,施主要是能够领会"空"的禅意,那离修成正果不远矣~~~ 
     在 Linux 系统里,有个设备档位于 /dev/null 。许多人都问过我那是甚么玩意儿?我跟你说好了:那就是"空"啦﹗没错﹗空空如也的空就是 null 了.... 请问施主是否忽然有所顿误了呢?然则恭喜了~~~ ^_^
    这个 null 在 I/O Redirection 中可有用得很呢:
* 若将 FD1 跟 FD2 转到 /dev/null 去,就可将 stdout 与 stderr 弄不见掉。
* 若将 FD0 接到 /dev/null 来,那就是读进 nothing 。
比方说,当我们在执行一个程序时,画面会同时送出stdout 跟 stderr,假如你不想看到 stderr (也不想存到档案去),那可以:
$ ls my.file no.such.file 2>/dev/null
my.file
     若要相反:只想看到 stderr 呢?还不简单﹗将 stdout 弄到 null 就行:
$ ls my.file no.such.file >/dev/null
ls: no.such.file: No such file or directory
    那接下来,假如单纯只跑程序,不想看到任何输出结果呢?哦,这里留了一手上次节目没讲的法子,专门赠予有缘人﹗... ^_^ 除了用 >/dev/null 2>&1 之外,你还可以如此:
$ ls my.file no.such.file &>/dev/null
(提示:将 &> 换成 >& 也行啦~~! )
    okay?讲完佛,接下来,再让我们看看如下情况:
$ echo "1" > file.out
$ cat file.out
1
$ echo "2" > file.out
$ cat file.out
2
    看来,我们在重导 stdout 或 stderr 进一份档案时,似乎永远只获得最后一次导入的结果。那,之前的内容呢?
    呵~~~ 要解决这个问提很简单啦,将 > 换成 >> 就好:
$ echo "3" >> file.out
$ cat file.out
2
3
    如此一来,被重导的目标档案之内容并不会失去,而新的内容则一直增加在最后面去。
    easy ? 呵 ... ^_^ 
    但是只要你再一次用回单一的 > 来重导的话,那么,旧的内容还是会被"洗"掉的﹗这时,你要如何避免呢?
----备份﹗ yes ,我听到了﹗不过.... 还有更好的吗?
    既然与施主这么有缘份,老纳就送你一个锦囊妙法吧:
$ set -o noclobber
$ echo "4" > file.out
-bash: file: cannot overwrite existing file 
    那么要如何取消这个"限制"呢?哦,将 set -o 换成 set +o 就行:
$ set +o noclobber
$ echo "5" > file.out
$ cat file.out
5
   再问:那... 有办法不取消而又"临时"盖写目标档案吗?
   哦,佛曰:不可告也﹗
   啊~~~ 开玩笑的、开玩笑的啦~~~ ^_^ 唉,早就料到人心是不足的了﹗
$ set -o noclobber
$ echo "6" >| file.out
$ cat file.out
6
   留意到没有:在 > 后面再加个" | "就好(注意: > 与 | 之间不能有空白哦)....
   呼.... (深呼吸吐纳一下吧)~~~ ^_^
   再来还有一个难题要你去参透的呢:
$ echo "some text here" > file
$ cat < file
some text here
$ cat < file > file.bak
$ cat < file.bak
some text here
$ cat < file > file
$ cat < file
    嗯?﹗注意到没有?﹗﹗
---- 怎么最后那个 cat 命令看到的 file 竟是空的?﹗
why? why? why?
    同学们:下节课不要迟到啰~~~!


--------------
11.5
    当当当~~~ 上课啰~~~ ^_^
    前面提到:$ cat < file > file 之后原本有内容的档案结果却被洗掉了﹗ 要理解这一现像其实不难,这只是 priority 的问题而已:
* 在 IO Redirection 中,stdout 与 stderr 的管道会先准备好,才会从 stdin 读进资料。
   也就是说,在上例中,> file 会先将 file 清空,然后才读进 < file,但这时候档案已经被清空了,因此就变成读不进任何资料了...
    哦~~~ 原来如此~~~~ ^_^
    那... 如下两例又如何呢?
$ cat <> file
$ cat < file >> file
   嗯... 同学们,这两个答案就当练习题啰,下节课之前请交作业﹗
    好了,I/O Redirection 也快讲完了,sorry,因为我也只知道这么多而已啦~~~ 嘻~~ ^_^ 不过,还有一样东东是一定要讲的,各位观众(请自行配乐~!#¥%¥%#@&……) :
---- 就是 pipe line 也﹗
    谈到 pipe line ,我相信不少人都不会陌生:
    我们在很多 command line 上常看到的" | "符号就是 pipe line 了。不过,究竟 pipe line 是甚么东东呢? 别急别急... 先查一下英汉字典,看看 pipe 是甚么意思?
   没错﹗它就是"水管"的意思...那么,你能想象一下水管是怎么一根接着一根的吗? 每根水管之间的 input 跟 output 又如何呢?
   嗯??
   灵光一闪:原来 pipe line 的 I/O 跟水管的 I/O 是一模一样的:
* 上一个命令的 stdout 接到下一个命令的 stdin 去了﹗
   的确如此... 不管在 command line 上你使用了多少个 pipe line ,
   前后两个 command 的 I/O 都是彼此连接的﹗(恭喜:你终于开窍了﹗ ^_^ )
    不过... 然而... 但是... ... stderr 呢?
   好问题﹗不过也容易理解:
* 若水管漏水怎么办?
    也就是说:在 pipe line 之间,前一个命令的 stderr 是不会接进下一命令的 stdin 的,其输出,若不用 2> 导到 file 去的话,它还是送到监视器上面来﹗这点请你在 pipe line 运用上务必要注意的。
     那,或许你又会问:
* 有办法将 stderr 也喂进下一个命令的 stdin 去吗?
(贪得无厌的家伙﹗)
方法当然是有,而且你早已学过了﹗ ^_^
我提示一下就好:
* 请问你如何将 stderr 合并进 stdout 一同输出呢?
   若你答不出来,下课之后再来问我吧... (如果你脸皮真够厚的话...)
   或许,你仍意尤未尽﹗或许,你曾经碰到过下面的问题:
* 在 cm1 | cm2 | cm3 ... 这段 pipe line 中,若要将 cm2 的结果存到某一档案呢?
    若你写成 cm1 | cm2 > file | cm3 的话,
    那你肯定会发现 cm3 的 stdin 是空的﹗(当然啦,你都将水管接到别的水池了﹗)
     聪明的你或许会如此解决:
cm1 | cm2 > file ; cm3 < file
    是的,你的确可以这样做,但最大的坏处是:这样一来,file I/O 会变双倍﹗
    在 command 执行的整个过程中,file I/O 是最常见的最大效能杀手。
    凡是有经验的 shell 操作者,都会尽量避免或降低 file I/O 的频率。
    那,上面问题还有更好方法吗?有的,那就是 tee 命令了。
* 所谓 tee 命令是在不影响原本 I/O 的情况下,将 stdout 复制一份到档案去。
   因此,上面的命令行可以如此打:
cm1 | cm2 | tee file | cm3
   在预设上,tee 会改写目标档案,若你要改为增加内容的话,那可用 -a 参数达成。
    基本上,pipe line 的应用在 shell 操作上是非常广泛的,尤其是在 text filtering 方面,凡举cat, more, head, tail, wc, expand, tr, grep, sed, awk, ... 等等文字处理工具,搭配起 pipe line 来使用,你会惊觉 command line 原来是活得如此精彩的﹗ 常让人有"众里寻他千百度,蓦然回首,那人却在灯火阑珊处﹗"之感... ^_^
....
    好了,关于 I/O Redirection 的介绍就到此告一段落。
    若日后有空的话,再为大家介绍其它在 shell 上好玩的东西﹗bye... ^_^

上一篇:十一问:>与<差别在哪(上篇) 下一篇:十二问:你要if还是case呢?

power by soyo123 2007-2008