《Unix & Linux 大学教程》 - 第十四、十五章 学习笔记

阅读更多

学习笔记,内容基础,适合初学者。

 

阅读之前,请务必花30秒查看前言说明(在第一、二章前面部分)

《Unix & Linux 大学教程》 - 第一、二章 学习笔记    Unix简介 & 什么是Linux?什么是Unix

《Unix & Linux 大学教程》 - 第三、四章 学习笔记    Unix连接 & 开始使用Unix

《Unix & Linux 大学教程》 - 第五、六章 学习笔记    GUI:图形用户界面 & Unix工作环境

《Unix & Linux 大学教程》 - 第七、八章 学习笔记    Unix键盘使用 & 能够立即使用的程序

《Unix & Linux 大学教程》 - 第九、十章 学习笔记    文档资料:Unix手册与Info & 命令语法

《Unix & Linux 大学教程》 - 第十一、十二章 学习笔记    shell & 使用shell:变量和选项

《Unix & Linux 大学教程》 - 第十三章 学习笔记    使用shell:命令和定制

《Unix & Linux 大学教程》 - 第十四、十五章 学习笔记     使用shell:初始化文件

《Unix & Linux 大学教程》 - 第十六、十七章 学习笔记    过滤器:简介和基本操作
《Unix & Linux 大学教程》 - 第十八章 学习笔记     过滤器:统计和格式化
《Unix & Linux 大学教程》 - 第十九章(一) 学习笔记     过滤器:选取、排序、组合及变换
《Unix & Linux 大学教程》 - 第十九章(二)、第二十章 学习笔记   过滤器:选取、排序、组合及变换 正则表达式

《Unix & Linux 大学教程》 - 第二十一章 学习笔记  显示文件

《Unix & Linux 大学教程》 - 第二十二章(一) 学习笔记  vi文本编辑器(一)

《Unix & Linux 大学教程》 - 第二十二章(二) 学习笔记  vi文本编辑器 (二)

《Unix & Linux 大学教程》 - 第二十二章(三) 学习笔记  vi文本编辑器 (三)

《Unix & Linux 大学教程》 - 第二十三章 学习笔记   Unix文件系统

《Unix & Linux 大学教程》 - 第二十四章 学习笔记  目录操作

《Unix & Linux 大学教程》 - 第二十五章 学习笔记  文件操作

《Unix & Linux 大学教程》 - 第二十六章(一) 学习笔记  进程和作业控制

《Unix & Linux 大学教程》 - 第二十六章(二) 学习笔记  进程和作业控制

《Unix & Linux 大学教程》 - 附录F 时区与24小时制时间

 

第十四章:使用shell:初始化文件

初始化文件(initialization file)

登录文件(login file):存放着所有希望在每次登录时自动执行的命令。

环境文件(environment file):存放着所有希望在新shell启动时自动执行的命令。

注销文件(logout file):存放注销时自动运行的命令。

 

shell 登录文件 环境文件 注销文件
C-Shell .login .cshrc .logout
Tcsh .login .tcshrc、.cshrc .logout
Bourne Shell .profile -- --
Korn Shell .profile $ENV --
Bash(默认) .bash_profile、.bash_login .bashrc .bash_logout
Bash(POSIX) .profile $ENV .bash_logout

注:

1.Korn Shell和Bash(POSIX)不使用标准的环境文件名称。可以将ENV变量设置为希望使用的任意文件的名称。

2.如果shell不支持注销文件,可以通过捕获EXIT信号使用注销文件。

 

POSIX标准要求shell应该支持登录文件和环境文件,没有要求注销文件。

如果默认不执行注销文件,则可以在登录文件中捕获EXIT信号来执行注销文件。

trap '. ${HOME}/.logout; exit' EXIT

登录文件名应为.profile,为了保持灵活性,环境文件名称没有固定而是保存到环境变量ENV中。

Bash可以以两种模式运行,默认模式(追求灵活性)和POSIX模式(追求兼容性)。

Bash支持POSIX标准的增强版,在POSIX模式中,Bash严格遵循1003.2标准。

 

POSIX模式中运行Bash

1.通用方法

bash --posix

2.部分适用(大多数Linux)

bash命令以默认方式启动,sh命令以POSIX模式启动

 

 

隐藏文件:点文件,文件名以点开头。

rc(run commands,运行命令)文件:文件名以点开头,以rc结尾。通常用于存放初始化命令。

ls -a:列举出所有文件,包括隐藏文件

 

登录shell:登录时启动的shell

非登录shell:其他的交互式shell

1.打开一个终端窗口,只是启动一个非登录shell,ctrl+alt+f1进行登录,则启动一个登录shell

2.任何时候都可以通过输入shell的名称启动一个新shell,新shell是一个非登录shell

3.使用ssh远程连接主机,必须登录,所以启动的是登录shell

 

登录shell执行登录文件和环境文件

非登录shell只执行环境文件

 

上面的规则也有特殊的情况,比如Bash默认模式

Bash(默认模式)

1.登录shell:.bash_profile

2.非登录shell:.bashrc

 

Bash(POSIX模式)

1.登录shell:.profile,然后$ENV

2.非登录shell:$ENV

 

 

Korn Shell

1.登录shell:.profile,然后$ENV

2.非登录shell:$ENV

 

(C-Shell部分在297页)

 

登录文件应该包含下述命令

1.创建或者修改环境变量的命令

2.执行所有一次性操作的命令

由于环境自动被所有子进程继承,所以环境变量只需要在登录文件中设置一次。在环境文件中设置环境变量是没有意义的,环境文件中设置的变量在新的shell启动时将被复位。

环境变量设置不能在环境中保存的自定义项,特别是shell选项、别名和函数。因为这些没有存储在环境中,所以每次启动新shell时必须重新创建他们。

 

 

名词解释:

CTSS(Compatible Time Sharing System):兼容分时系统。

 

 

 

 

第十五章:标准I/O:重定向和管道

 

标准输入:stdin

标准输出:stdout

标准错误:stderr

标准I/O:上述三者总称

 

重定向标准输出、标准输入

>>:可以将输出重定向到文件,但是是追加到文件末尾

>:可以将输出重定向到文件,文件不存在则建立,如果存在,则删除之前内容。

<:可以将输入从键盘重定向到文件等。

sort < 1.txt > report

上面例子中, sort先从1.txt中读取数据,排序之后输出到report文件中。

 

shell选项:noclobber

开启此选项可以防止文件被重定向替换或创建。

$ ls > 1.txt
$ ls > 1.txt
bash: 1.txt: cannot overwrite existing file

我连续两次将ls输出重定向到1.txt,原始状态是没有1.txt这个文件的,第一个命令创建了这个文件,再此执行的时候,由于设置了noclobber=on(set -o noclobber),所以会提示错误。

如果依然想执行这条命令,将1.txt的内容替换掉,那么可以使用:

ls >| 1.txt

这样就能强制输出到1.txt文件中。(C-Shell使用ls >! 1.txt)

Bourne shell中,noclobber不会影响“>>”(C-Shell会影响,如果指定的文件不存在,则会出现错误提示,可以通过“ls >>! 1.txt”临时执行)。

 

文件描述符(file descriptor)

Unix进程中,每个输入源和每个输出目标都有一个唯一的数字标识,这个数字称为文件描述符。

在编写程序时,使用文件描述符控制I/O,每个文件使用一个文件描述符。

 

Bourne Shell家族中,重定向输入或输出的正式语法是:在文件描述符数字之后使用小于号或大于号。

calculate 8> results

表示calculate将输出写入到文件描述符为8的results中。

 

Unix为每个进程提供3个预定义的文件描述符。

标准输入:0

标准输出:1

标准错误:2

只能省略标准输入和标准输入的文件描述符,标准错误不能省略。

calculate 0< results同calculate < results

calculate 1> results同calculate > results

calculate 2> results不能省略

 

进程就是加载到内存中并且准备运行的程序,以及程序的数据和跟踪程序所需的信息。当进程需要启动另一个进程时,这个进程就创建一个副本进程。原始进程称为父进程,副本进程称为子进程。

子进程开始运行后,父进程等待子进程死亡(结束)。一旦子进程死亡,父进程就会被唤醒,重新获得控制权并再次开始运行,此时子进程消失。

 

当创建子shell时,它继承父shell的环境。但是子shell对环境的修改不会传递到父shell。

两种方式创建子shell

1.输入shell名启动shell,^D后退出。

2.将命令用圆括号括起来,临时使用子shell执行。位于圆括号中的这些命令被称为一个编组(grouping)

使用编组和子shell的最常见的原因是防止cd命令影响当前的shell,其通常格式为:

(cd directorycommand)

 

组合标准输出和标准错误

下面将标准输出和标准错误输出到同一个文件中,语法如下

command x> outputfile y>&(这里x和y是文件描述符)

例如,ls filename,如果存在此file,那么终端则显示,如果没有,终端则输出错误提示。

ls: cannot access a: No such file or directory

现在在我的工作目录下,有一个文件为1.txt,但是没有2.txt

执行:ls 1.txt 2.txt

则输出

ls: cannot access 2.txt: No such file or directory
1.txt

如果执行:ls 1.txt 2.txt > 1.txt

则在终端只显示

ls: cannot access 2.txt: No such file or directory

打开1.txt则只有一行为:

1.txt

改用下面的命令即可将标准输出和标准错误都重定向写入到1.txt中

ls 1.txt 2.txt > 1.txt 2>&1

当然也可以使用:ls 1.txt 2.txt 1> 1.txt 2>&1

当然,也可以先考虑标准错误,使用:ls 1.txt 2.txt 2> 1.txt 1>&2

 

对于Bash来说,可以使用下面命达到同样效果:

ls 1.txt 2.txt &> 1.txt

ls 1.txt 2.txt >& 1.txt

但是如果想把输出追加到文件末尾,那么还得使用之前的方法:

ls 1.txt 2.txt >> 1.txt 2>&1

 

*注意区分:ls 1.txt 2.txt 2>&1 > 1.txt

上面的命令意思为:先将标准错误重定向到与标准输出同一个位置,但是此时标准输出还没有被重定向,标准输出发送给了shell,所以导致标准错误也发送给了shell。之后的事情大家就都知道了,重定向了标准输出。最后结果,终端会显示标准错误,标准输出被重定向到了1.txt中。

 

如果一条命令中,向同一个文件重定向两次的话,一个重定向就会覆盖另一个。所以下面的命令不能正常工作。

ls 1.txt 2.txt > 1.txt 2> 1.txt

 

抛弃输出与错误

位桶(bit bucket):/dev/null

发给/dev/null的任何东西都会消失。

所以我们如果不想看到标准输出,可以这样:

ls 1.txt 2.txt > /dev/null

如果不想看到标准错误:

ls 1.txt 2.txt 2> /dev/null

如果都不想看到:

ls 1.txt 2.txt > /dev/null 2>&1

 

Bourne shell重定向元字符总结

元字符 动作
< 重定向标准输入(同0<)
> 重定向标准输出(同1>)
>| 重定向标准输出:强制重写
>> 追加标准输出(同1>>)
2> 重定向标准错误
2>> 追加标准错误
2&>1 将标准错误重定向到标准输出
>&或者&> 重定向标准输出+标准错误(只适用于Bash)
| 将标准输出通过管道传送给令一条命令
2>&1| 将标准输出+标准错误通过管道传送给令一条命令

管道线

shell允许创建一序列命令,在这一序列命令中,一个程序的标准输出可以自动的发送给下一个程序的标准输入。两个程序之间的连接就是管道(pipe),而命令序列本身称为管道线(pipeline)

Bourne shell可以将标准输出和标准错误组合在一起发送给另一个程序:

command1 2>&1 | command2

C-Shell版为:

command1 |& command2

 

当创建管道线时,必须使用能够从标准输入读取文本,并向标准输出写入文本的程序。这样的程序称为过滤器

 

管道线分流

tee [-afile...

tee从标准输入读取数据,并向文件和标准输出各发送一份数据。

ls | tee 1.txt

这样就能将ls的输出显示到屏幕上面并且同时保存到1.txt中一份。

如果tee后面的文件参数不存在,则自动创建一份,如果存在,则覆盖旧文件,可以使用-a(append)改为追加。

 

条件执行

伪代码:

if(action1()){
    action2();
}

if(!action1()){
    action2();
}

上面两段代码在shell中分辨对应

command1 && command2

command1 || command2

如果grep成功,则在标准输出显示结果,如果grep失败,则不显示任何内容。

现在我有一个名text.txt的文本文件,里面只有一行字符串:this is a test

现在查找this单词,如果存在,则显示success

grep this text.txt && echo "success"

现在查找these单词,如果不存在,则显示fail

grep these text.txt || echo "fail"

我们发现,当grep成功的时候,会把满足条件行显示在屏幕上,如果不想显示,我们可以考虑将标准输出重定向到/dev/null中。

grep this text.txt > /dev/null && echo "success"

 

如果想实现下面的功能,应该如何做呢

if(action1()){
    action2();
}else{
    action3();
}

书中没有给出答案,下面是我的想法

 

grep word text.txt && echo "success" || echo "fail"

如果grep word text.txt成功了,那么应该执行echo "success",所以echo "success"也成功了,那么后面的||就不会执行了。

如果grep word text.txt失败了,那么echo "success"不会执行,所以echo "success"失败了,那么后面的||应该执行。

不要想着使用括号提高运算级别,使用括号会启动子shell。

 

 

 

 

转贴请保留以下链接

本人blog地址

http://su1216.iteye.com/

http://blog.csdn.net/su1216/

你可能感兴趣的:(管道,分流,标准输入,标准输出,标准错误)