Linux Shell 常用操作与脚本示例详解
$[] 符号的运算
$[]
是 Shell 中用于整数运算的符号,支持加减乘除、取余及自增自减等操作
示例 1:计算 1 到 10 的总和
[lyk@controller bin 15:37:04]$ echo {1..10}1 2 3 4 5 6 7 8 9 10[lyk@controller bin 16:00:54]$ echo {1..10} | sed 's/ /+/g'1+2+3+4+5+6+7+8+9+10[lyk@controller bin 16:01:58]$ echo $(echo {1..10} | sed 's/ /+/g')=1+2+3+4+5+6+7+8+9+10=[lyk@controller bin 16:02:07]$ echo $(echo {1..10} | sed 's/ /+/g')=$(echo $[ $(echo {1..10}|sed 's/ /+/g')])1+2+3+4+5+6+7+8+9+10=55
示例 2:基础算术运算
[lyk@controller bin 16:03:32]$ echo $[1+1]2[lyk@controller bin 16:04:31]$ echo $[4-2]2[lyk@controller bin 16:04:36]$ echo $[2*3]6[lyk@controller bin 16:04:54]$ echo $[4/1]4[lyk@controller bin 16:04:58]$ echo $[5/2]2[lyk@controller bin 16:05:13]$ echo $[5%2]1[lyk@controller bin 16:05:19]$ count=3;echo $[(count+1)*3] # (3+1)*3=1212[lyk@controller bin 16:05:36]$ count=3;echo $[ ++count + 3 ]# count先变成4,4+3=77[lyk@controller bin 16:05:41]$ count=3;echo $[ count++ + 3 ]# 先算3+3=6,count再变成46[lyk@controller bin 16:05:54]$ count=3;echo $[ --count + 3 ]# count先变成2,2+3=55[lyk@controller bin 16:06:00]$ count=3;echo $[ count-- + 3 ]# 先算3+3=6,count再变成26
案例1:统计/etc/profile文件中每个单词出现的次数,并显示数量最多的10个单词
以下三种方法均用于统计
/etc/profile
中每个单词的出现次数,并显示出现次数最多的前 10 个单词(方法差异主要在 “提取单词” 的方式)
#方法一:用 egrep 提取纯字母单词[lyk@controller bin 16:06:05]$ cat /etc/profile| egrep -o '[a-zA-Z]+' | sort | uniq -c | sort -nr|head12 usr8 then8 if8 fi7 bin6 pathmunge6 PATH6 in6 id6 icat /etc/profile:读取文件内容。egrep -o '[a-zA-Z]+':用正则提取所有纯字母单词(-o 只输出匹配的部分,[a-zA-Z]+ 匹配连续字母)。sort:对单词排序(为去重做准备)。uniq -c:统计每个单词的出现次数(-c 前缀显示次数)。sort -nr:按次数从高到低排序(-n 按数字排序,-r 逆序)。head:默认显示前 10 行。#方法二:用 tr 分割单词,sed 取范围[lyk@controller bin 16:31:35]$ cat /etc/profile | tr '[[:punct:] ]' '\n' | sort | uniq -c | sort -nr | sed -n '2,11p'12 usr8 then8 if8 fi7 bin6 pathmunge6 PATH6 in6 id6 itr '[[:punct:] ]' '\n':将所有标点符号([:punct:])和空格替换为换行符,实现按标点 / 空格拆分单词sed -n '2,11p':只显示第 2 到 11 行(跳过出现次数最多的第 1 行,取接下来的 10 行)#方法三:用 tr 分割单词,awk 取范围[lyk@controller bin 16:31:38]$ cat /etc/profile | tr '[[:punct:] ]' '\n' | sort | uniq -c | sort -nr|awk 'NR>=2 && NR<=11'12 usr8 then8 if8 fi7 bin6 pathmunge6 PATH6 in6 id6 iawk 'NR>=2 && NR<=11':用 awk 筛选行号(NR 表示行号),只保留第 2 到 11 行,效果同方法二的 sed
案例2:文件 / 目录判断与操作
利用 Shell 的条件判断(
[ ]
)可检查文件 / 目录属性,并执行相应操作
示例 1:判断文件是否可读取
# 检查是否有 /etc/shadow 的读权限,有则输出提示[lyk@controller ~ 16:51:39]$ [ -r /etc/shadow ] && echo i can read /etc/shadow# (无输出,因为普通用户没有 /etc/shadow 的读权限)# 检查是否没有读权限,没有则输出提示[lyk@controller ~ 16:51:56]$ [ ! -r /etc/shadow ] && echo i can not read /etc/shadowi can not read /etc/shadow# 普通用户无权限,故输出-- [ -r 文件名 ]:判断文件是否有读权限-- !:取反(! -r 表示 “没有读权限”)-- &&:逻辑与(前面条件成立时,执行后面的命令
示例 2:判断目录是否存在并操作
# 假设 $path 是 /tmp/data# 检查目录是否存在,存在则提示[lyk@controller ~ 16:53:06]$ [ -d $path ] && echo $path is a directory# (无输出,因为目录不存在)# 检查目录是否不存在,不存在则提示[lyk@controller ~ 16:53:44]$ [ ! -d $path ] && echo $path is not a directory/tmp/data is not a directory# 目录不存在则创建,并提示[lyk@controller ~ 16:54:03]$ [ ! -d $path ] && mkdir $path && echo 创建$path成功创建/tmp/data成功# 目录存在则删除[lyk@controller ~ 16:56:35]$ [ -d $path ] && rmdir $path# 检查目录是否已删除[lyk@controller ~ 16:57:12]$ ls $pathls: 无法访问/tmp/data: 没有那个文件或目录[lyk@controller ~ 16:57:19]$ [ -d $path ] && echo $path is exist# (无输出,因为不存在)# 重新创建目录后检查[lyk@controller ~ 16:58:16]$ mkdir $path[lyk@controller ~ 16:58:29]$ [ -d $path ] && echo $path is exist/tmp/data is exist[ -d 目录名 ]:判断是否为目录且存在。mkdir $path:创建目录。rmdir $path:删除空目录。
案例3:判断当前用户是否是root用户
以下四种方法均用于检查执行脚本的用户是否为 root,若不是则提示并退出
#方法 1:用 whoami 命令判断[lyk@controller bin 17:24:26]$ vim 123.sh1 #!/bin/bash 2 [ "$(whoami)" != "root" ] && echo pls run as root. && exit[lyk@controller ~ 17:23:34]$ cd bin[lyk@controller bin 17:24:00]$ mv ~/123.sh .[lyk@controller bin 17:24:21]$ 123.sh[lyk@controller ~ 17:22:14]$ chmod +x 123.sh[lyk@controller bin 17:24:21]$ 123.shpls run as root.-- $(whoami):执行 whoami 命令,获取当前用户名(root 用户返回 root)。-- [ "$(whoami)" != "root" ]:若用户名不是 root,则条件成立。-- && exit:提示后退出脚本。
#方法 2:用 $USER 变量判断[lyk@controller bin 17:25:42]$ vim 123.sh1 #!/bin/bash2 [ "$USER" != "root" ] && echo pls run as root. && exit [lyk@controller bin 17:27:35]$ 123.shpls run as root.-- $USER:Shell 内置变量,直接表示当前用户名(root 用户为 root)
#方法 3:用 $UID 变量判断[lyk@controller bin 17:28:28]$ vim 123.sh1 #!/bin/bash2 [ "$UID" -ne 0 ] && echo pls run as root. && exit [lyk@controller bin 17:28:27]$ 123.shpls run as root.-- $UID:Shell 内置变量,表示当前用户的 UID(root 用户的 UID 是 0,普通用户 UID ≥ 1000)-- -ne 0:不等于 0(即非 root 用户)
#方法 4:用 (()) 算术表达式判断 [lyk@controller bin 17:28:57]$ vim 123.sh1 #!/bin/bash 2 ((UID!=0)) && echo pls run as root. && exit [lyk@controller bin 17:29:44]$ 123.sh pls run as root. -- ((UID!=0)):算术表达式写法,判断 UID 是否不等于 0(非 root 用户),更简洁