仅仅是学习Linux系统的命令还不够,只有把多个命令按照自己想要的方式进行组合使用,才能提高工作效率。今天的内容主要是关于如何把命令组合在一起使用,使得输入的命令更准确、更高效,也为接下来的Shell脚本打好基础。 一、输入输出重定向输入重定向:指把文件内容导入到命令中。 输出重定向:指把原本要输出到屏幕的数据信息写入到指定的文件中,又分为标准输出重定向和错误输出重定向两种。
要区别对待两种输出信息,第一种是命令的标准输出信息,第二种是命令的报错提示信息(错误输出),如下所示: [root@linuxprobe ~]# ls -l anaconda-ks.cfg
-rw-------. 1 root root 1032 Feb 18 2019 anaconda-ks.cfg //ls命令的标准输出信息,也是我们想要的信息
[root@linuxprobe ~]#
[root@linuxprobe ~]# ls -l xxxxx
ls: cannot access xxxxx: No such file or directory //因为xxxxx文件不存在,所以输出的是报错提示信息
[root@linuxprobe ~]#
对于输入重定向来说,用到的符号及其作用如下所示:
对于输出重定向来说,用到的符号及其作用如下所示:
通过标准输出重定向将 man ls 命令原本要输出到屏幕的信息写入文件readme.txt中,然后显示readme.txt文件中的内容,具体命令如下: [root@linuxprobe ~]# man ls > readme.txt
[root@linuxprobe ~]#
[root@linuxprobe ~]# cat readme.txt
LS(1) User Commands LS(1)
NAME
ls - list directory contents
SYNOPSIS
ls [OPTION]... [FILE]...
DESCRIPTION
List information about the FILEs (the current directory by default). Sort entries alphabetically if none of -cftuvSUX nor
--sort is specified.
---------------------------------省略部分输出内容---------------------
通过覆盖写入模式向readme.txt文件中写入一行数据,然后再通过追加写入模式再写入一行数据,具体命令如下: [root@linuxprobe ~]# echo "Welcome to my home" > readme.txt //清除原有的内容
[root@linuxprobe ~]# echo "Learning Linux is happy to me" >> readme.txt //追加至原来文件内容的后面
[root@linuxprobe ~]#
[root@linuxprobe ~]# cat readme.txt
Welcome to my home
Learning Linux is happy to me
[root@linuxprobe ~]#
如果想把命令的报错信息写入文件,该如何操作呢? [root@linuxprobe ~]# ls -l xxxxx
ls: cannot access xxxxx: No such file or directory //提示xxxxx文件不存在
[root@linuxprobe ~]#
[root@linuxprobe ~]# ls -l xxxxx 2> readme.txt //将报错信息写入readme.txt文件
[root@linuxprobe ~]#
[root@linuxprobe ~]# cat readme.txt
ls: cannot access xxxxx: No such file or directory //报错信息写入readme.txt文件成功
[root@linuxprobe ~]#
输入重定向相对来说比较冷门,在工作中遇到的概率较小。输入重定向的作用是把文件内容直接导入命令中。接下来使用输入重定向把readme.txt文件导入给wc -l命令,统计文件内容的行数。 [root@linuxprobe ~]# cat readme.txt
ls: cannot access xxxxx: No such file or directory
[root@linuxprobe ~]#
[root@linuxprobe ~]# wc -l < readme.txt //使用输入重定向
1
[root@linuxprobe ~]#
二、管道命令符管道符,即“|”,其执行的格式“命令A | 命令B”,当然可以这样使用:“命令A | 命令B | 命令C”。管道命令符的作用是把前一个命令A原本要输出到屏幕的标准正常数据当作是后一个命令B的标准输入,比如把搜索命令的输出值传递给统计命令,具体如下: [root@localhost ~]# grep "/sbin/nologin" /etc/passwd | wc -l
19
[root@localhost ~]#
再比如用翻页的形式查看/etc目录中的文件列表及属性信息: [root@localhost ~]# ls -l /etc/ | more
total 1168
-rw-r--r--. 1 root root 16 Jun 18 2019 adjtime
-rw-r--r--. 1 root root 1518 Jun 7 2013 aliases
-rw-r--r--. 1 root root 12288 Jun 18 2019 aliases.db
drwxr-xr-x. 2 root root 236 Jun 18 2019 alternatives
-rw-------. 1 root root 541 Mar 31 2016 anacrontab
-rw-r--r--. 1 root root 55 Nov 4 2016 asound.conf
-rw-r--r--. 1 root root 1 Oct 30 2018 at.deny
drwxr-x---. 3 root root 43 Jun 18 2019 audisp
drwxr-x---. 3 root root 83 Jun 18 2019 audit
drwxr-xr-x. 2 root root 22 Jun 18 2019 bash_completion.d
-rw-r--r--. 1 root root 2853 Nov 5 2016 bashrc
drwxr-xr-x. 2 root root 6 Nov 6 2016 binfmt.d
-rw-r--r--. 1 root root 38 Nov 29 2016 centos-release
--More--
在修改用户密码时,通常都需要输入两次密码以进行确认,这在编写自动化脚本时将成为一个非常致命的缺陷。通过管道符和passwd命令的--stdin参数相结合,可以用一条命令来完成密码重置的操作: [root@localhost ~]# echo "123456" | passwd --stdin root //一条命令修改root用户密码
Changing password for user root.
passwd: all authentication tokens updated successfully.
[root@localhost ~]#
通过重定向技术能够一次性地把多行信息打包输入或输出,比如让用户一直输入内容,直到用户输入了其自定义的分界符时,才结束输入。这种方法在编写自动化脚本时经常用到。 [root@localhost ~]# mail -s "readme" root@localhost << over //"over"为用户自定义地分界符
> i think linux is very practical
> i hope to learn more
> can you teach me?
> over //遇到分界符后结束输入,以上3行信息则为用户输入的有效内容
[root@localhost ~]#
三、命令行的通配符通配符,顾名思义,就是通用的匹配信息的符号,常见的通配符如下所示:
匹配在/dev目录中所有以sda开头的文件: [root@localhost ~]# ls -l /dev/sda*
brw-rw----. 1 root disk 8, 0 Feb 20 23:57 /dev/sda
brw-rw----. 1 root disk 8, 1 Feb 20 23:57 /dev/sda1
brw-rw----. 1 root disk 8, 2 Feb 20 23:57 /dev/sda2
[root@localhost ~]#
匹配以sda开头,且后面还紧跟某一个字符(字符包括字母、数字、特殊符号等等)的文件: [root@localhost ~]# ls -l /dev/sda? //该命令排除了/dev/sda文件,因为不匹配空值
-rw-r--r--. 1 root root 0 Feb 21 02:39 /dev/sda@ //该文件为本人所创建,用来实验测试
brw-rw----. 1 root disk 8, 1 Feb 20 23:57 /dev/sda1
brw-rw----. 1 root disk 8, 2 Feb 20 23:57 /dev/sda2
[root@localhost ~]#
匹配以sda开头,且后面还紧跟某一个数字的文件: [root@localhost ~]# ls -l /dev/sda[0-9]
brw-rw----. 1 root disk 8, 1 Feb 20 23:57 /dev/sda1
brw-rw----. 1 root disk 8, 2 Feb 20 23:57 /dev/sda2
[root@localhost ~]#
匹配以sda开头,且后面还紧跟1、3、5中的某一个数字的文件: [root@localhost ~]# ls -l /dev/sda[135] //最好写成[1,3,5],因为这种写法更规范
brw-rw----. 1 root disk 8, 1 Feb 20 23:57 /dev/sda1
[root@localhost ~]#
四、常用的转义字符Shell解释器提供了丰富的转义字符来处理输入的特殊数据,常见的4个转义字符如下所示:
实验:先定义一个名为PRICE的变量并赋值5,然后输出以双引号括起来的字符串与变量信息: [root@localhost ~]# PRICE=5 [root@localhost ~]# [root@localhost ~]# echo "Price is $PRICE" //保留其中变量属性 Price is 5 实验:想要输出“Price is $5”,需要使用反斜杠进行转义。 [root@localhost ~]# echo "Price is $$PRICE" //$$的作用是显示当前程序的进程ID号码
Price is 2571PRICE
[root@localhost ~]#
[root@localhost ~]# echo "Price is \$$PRICE" //使用反斜杠进行转义
Price is $5
[root@localhost ~]#
实验:输出命令执行的结果。 [root@localhost ~]# echo `uname -a` //先执行uname -a命令,然后输出执行结果
Linux localhost.localdomain 3.10.0-514.el7.x86_64 #1 SMP Tue Nov 22 16:42:41 UTC 2016 x86_64 x86_64 x86_64 GNU/Linux
[root@localhost ~]#
五、重要的环境变量变量是计算机系统用于保存可变值的数据类型。在Linux系统中,变量名称一般都用大写表示,这是一种约定俗成的规范。可以直接通过变量名称来提取对应的变量值。Linux系统中的环境变量是用来定义系统运行环境的一些参数,比如每个用户的家目录、邮件存放的位置等。 在执行了一条命令后,Linux系统中到底发生了什么呢?简单来说,主要分为4个步骤: 第1步:判断用户是否以绝对路径或相对路径的方式输入命令(如/bin/ls就是以绝对路径的方式执行命令),如果是的话则直接执行。 第2步:Linux系统检查用户输入的命令是否为“别名命名”,即用一个自定义的命令名称来替换原本的命令名称。alias命令可以用来创建属于自己的命令别名,格式为“alias 别名=命令”。unalias命令用来取消命令别名,格式为“unalias 别名”。在上图中,输入ls命令后,不同的文件类型显示不同的颜色,这其实就是Linux系统为了方便用户区分文件类型而特意设置的ls命令别名。 [root@localhost ~]# alias ls='ls --color=auto' //设置命令的别名,执行ls命令则等同于执行ls --cloar=auto
[root@localhost ~]#
第3步:Bash解释器判断用户输入的是内部命令还是外部命令。内部命令是解释器内部的命令,会被直接执行。而用户在绝大多数输入的是外部命令,这些外部命令交由第4步处理。可以用“tyep 命令名称”来判断输入的命令是内部命令还是外部命令,能定位到命令存放的路径的都是外部命令。 [root@linuxprobe ~]# type cat
cat is /usr/bin/cat
[root@linuxprobe ~]# type more
more is /usr/bin/more
第4步:系统在多个路径中查找用户输入的命令文件,而定义这些路径的变量叫作PATH,作用是告诉Bash解释器将要执行的命令可能存放的位置,然后Bash解释器就会乖乖地在这些路径下逐个查找。PATH变量包含多个路径值,每个路径值之间用冒号间隔,对这些路径的增加和删除操作将影响到Bash解释器对Linux命令的查找。 [root@linuxprobe ~]# echo $PATH
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin
[root@linuxprobe ~]#
[root@linuxprobe ~]# PATH=$PATH:/root/bin //添加新的路径,不过系统重启后失效
[root@linuxprobe ~]#
[root@linuxprobe ~]# echo $PATH
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin:/root/bin //添加成功
[root@linuxprobe ~]#
综上所述,作为一名态度谨慎的运维工作者来说,在接手一台新的Linux系统后一定要检查PATH变量中是否有可疑的目录。另外,可以使用env命令来查看Linux系统中所有的环境变量,其中最重要的10个环境变量如下所示:
Linux作为一个多用户多任务的操作系统,能够为每个用户提供独立的、合适的工作运行环境,因此,一个相同的变量会因为用户身份的不同而具有不同的值。 [root@linuxprobe ~]# echo $HOME
/root
[root@linuxprobe ~]# su - linuxprobe //切换至linuxprobe用户
Last login: Sat Feb 15 19:28:26 BNT 2020 on :0
[linuxprobe@linuxprobe ~]$
[linuxprobe@linuxprobe ~]$ echo $HOME
/home/linuxprobe
[linuxprobe@linuxprobe ~]$
我们完全可以自行创建一个变量,来满足工作需求。例如设置一个名为WORKDIR的变量,如下所示: [root@linuxprobe ~]# mkdir /home/workdir //新建一个目录 [root@linuxprobe ~]# [root@linuxprobe ~]# [root@linuxprobe ~]# WORKDIR=/home/workdir //新建一个变量,并将路径赋值给WORKDIR变量 [root@linuxprobe ~]# [root@linuxprobe ~]# cd $WORKDIR [root@linuxprobe workdir]# pwd /home/workdir [root@linuxprobe workdir]# 这样的变量不具有全局性,作用范围有限,可以使用export命令将其提升为全局变量,这样其他用户就可以使用了。注意:当使用su - 命令切换用户时,export命令无效 [root@linuxprobe ~]#
[root@linuxprobe ~]# WORKDIR=/home/workdir/ //给WORKDIR变量赋值
[root@linuxprobe ~]# echo $WORKDIR
/home/workdir/
[root@linuxprobe ~]# export WORKDIR //提升为全局变量
[root@linuxprobe ~]#
[root@linuxprobe ~]# su linuxprobe //切换至linuxprobe用户
[linuxprobe@linuxprobe root]$
[linuxprobe@linuxprobe root]$ echo $WORKDIR //成功输出WORKDIR变量的值
/home/workdir/
[linuxprobe@linuxprobe root]$ su - linuxprobe //使用su -命令切换至linuxprobe用户
Password:
Last login: Sat Feb 22 11:52:23 BNT 2020 on pts/0
[linuxprobe@linuxprobe ~]$ echo $WORKDIR //无法查看WORKDIR变量的值
[linuxprobe@linuxprobe ~]$
|
|