1/35
bash
linux高级命令:
grep cut wc sort tr uniq diff patch which locate find stat
shell变量
shell类型
bash特性
for

技术回顾
高级命令:
切:cut 排序:sort 合并:uniq
统计:wc 替换:tr
显示文件状态信息(详细属性):stat
读文件内容:
读全文:cat 读开头:head 读结尾:tail
读更多(分页):more 读更少(分页):less
显示当前工作路径(等同于win7窗口中的地址栏):pwd
切换目录(即打开文件夹):cd 路径
读文件名列表:ls [-ld] [路径]
搜索:which whereis find locate(updatedb)
显示磁盘空间使用状态:df -h
显示进程信息:ps -elf
打包压缩:tar -czvf 包名.tar.gz 要打包的文件名
解包解压:tar -xzvf 包名.tar.gz [-C /目标路径]
scp命令:scp -rv 源 目标
远程发送文件给对方:scp -r /etc [email protected]:/opt/
远程从对方下载文件:scp -r [email protected]:/opt/ /tmp/
grep
grep命令:匹配关键字,即显示文件中关键字所在的整行内容。
grep root /etc/passwd 显示/etc/passwd文件中所有包含root关键字的行
grep -n root /etc/passwd 显示/etc/passwd文件所有包含root关键字的行并显示关键字处于文本中哪一

2/35
-n 显示行号
-i 不区分大小写
-v 取反
-c 带关键字的行数
-q 静默执行。即不在屏幕显示匹配到的内容。通常用于写shell脚本时。
-A 连带显示关键字之后的几行
grep -A 2 u1 /etc/passwd
u1:x:502:502::/home/u1:/bin/bash
u3:x:503:503::/home/u3:/bin/bash
u4:x:504:504::/home/u4:/bin/bash
-B 连带显示关键字之前的几行
grep -B 2 u1 /etc/passwd
tcpdump:x:72:72::/:/sbin/nologin
u2:x:501:501::/home/u2:/bin/bash
u1:x:502:502::/home/u1:/bin/bash
--color 带颜色显示关键字
grep root /etc/passwd --color
root:x:0:0:root:/root:/bin/bash
operator:x:11:0:operator:/root:/sbin/nologin
显示/etc/passwd文件中包含ssh关键字的行的用户的uid和gid
grep ssh /tmp/passwd |cut -d : -f 3,4
cut
cut 命令:提取文件内容中指定的字段(列)、字符内容。单词的默认分隔符为空格。
cut [选项]... [文件]...
选项:
-b 按照字节切,utf-8编码中每个汉字占3个字节,例如 4个汉字就得取12个字节。
-c 按照字符切
-d 指定分割符
-f 指定以分割符分割的哪个字段
head -5 /etc/passwd > /tmp/pass 将passwd文件中的前5行覆盖写入/tmp/pass文件中
cat /tmp/pass 显示/tmp/pass的全文内容
cut -d : -f 6 /tmp/pass 显示/tmp/pass文件中以:分隔的第6列的内容,家目录内容
cut -d : -f 1,7 /tmp/pass 显示/tmp/passwd文件中的第1列(用户名)和第7列(shell解释器)的内容
cut -d : -f 1,4-7 /tmp/pass 取1 4 5 6 7列
cut -d : -f -3 /tmp/pass 取第1列至第3列
cut -d : -f 3- /tmp/pass 取第3列至最后一列
cut -c 1-3 /tmp/pass 取每行的前3个字符
echo '×××' | cut -b 1-12 取1-12个字节
3/35
seq
seq:按顺序打印一些数字。如1 2 3 4 5 ...
用法:seq [选项]... 尾数
 或:seq [选项]... 首数 尾数
 或:seq [选项]... 首数 增量 尾数
例:依次执行如下的seq命令,熟悉seq的用法。
seq 10
seq 5 11
seq 1 2 10 输出1~10之间的奇数
seq 2 2 10 输出2~10之间的偶数
seq 10 -1 0 输出10~0之间的数
seq -w 1 50 200
用以下方法可以实现根seq同样的效果echo {首数..尾数..增量}
echo {1..10}
echo {5..11}
echo {1..10..2}
echo {0..10..2}
echo {10..0..-1}
sort
sort 排序(升序、降序)默认是按升序排列。
-n, --numeric-sort 根据数值比较,默认是按单个字符来比较
-r, --reverse 逆序输出排序结果
-t 指定分割符(separator)
-k 排序键(key,即排序的列字段)
-u 去掉重复行,u是uniq唯一的缩写
例:执行如下命令,熟悉sort排序的基本用法。
seq 1 15
seq 1 15 | sort
seq 1 15 | sort -n
seq 1 15 | sort -nr
准备阶段:用cat命令创建一个a.txt文件。
说明:以下是用cat命令将EOF...EOF(End Of File)之间的内容覆盖写入(>)到a.txt文件中
cat > a.txt <root:1:0:bin
bin:2:200:bash
jim:10:300:bash
tom:102:10:bash
EOF
4/35
查看a.txt文件内容:cat a.txt
sort a.txt //默认顺序排序(按第1列字符排升序)
bin:2:200:bash
jim:10:300:bash
root:1:0:bin
tom:102:10:bash
sort -r a.txt //逆序输出排序结果(按第1列字符排降序)
tom:102:10:bash
root:1:0:bin
jim:10:300:bash
bin:2:200:bash
sort -t : -k 2 a.txt //按:分割开的以第2部分的第一个字符排序
tom:102:10:bash
jim:10:300:bash
root:1:0:bin
bin:2:200:bash
sort -t : -k 2 -n a.txt //按:分割开的以第2列的数值大小排序(用得最多)
root:1:0:bin
bin:2:200:bash
jim:10:300:bash
tom:102:10:bash
sort -t : -k 1.2 a.txt 按:分割开的以第1列的第2个字符排序(很少用)
bin:2:200:bash
jim:10:300:bash
root:1:0:bin
tom:102:10:bash
将/etc/passwd 文件中第3列按数值大小排降序
sort -n -r -t : -k 3 /etc/passwd

echo "bin:2:200:bash" >> a.txt 将echo命令后的文本追加到a.txt文件中

cat a.txt 显示a.txt全文内容

root:1:0:bin
bin:2:200:bash
jim:10:300:bash
tom:102:10:bash
bin:2:200:bash

sort a.txt 将a.txt排序输出

bin:2:200:bash
bin:2:200:bash
jim:10:300:bash
root:1:0:bin
tom:102:10:bash

sort -u a.txt 去掉重复行

bin:2:200:bash
jim:10:300:bash
5/35
root:1:0:bin
tom:102:10:bash
1.查看/etc/passwd文件的前2行内容。然后取出本系统中所有的用户名、uid、用户的shell信息将结果保存
在/tmp/users.txt文件中?最后用cat查看users.txt文件。
head -2 /etc/passwd 查看前2行
cut -d : -f 1,3,7 /etc/passwd > /tmp/users.txt
cat /tmp/users.txt
说明:-d 指定分隔符,-f 指定字段(即列)
2.统计系统中有多少个用户?
whatis wc 显示命令功能简介
wc /etc/passwd 统计passwd文件中的行数、单词数、字符数
wc -l /etc/passwd 统计行数,即用户数量(每行一个用户账号)
说明:wc是统计单词数量、行数、字符数。-l 是统计行数,-c是统计字符数,-w显示单词计数
3.统计系统中有多少个用户账号是允许登录?
grep -E 'bash$|sh$' /etc/passwd | wc -l
4.统计系统中有多少个用户账号不允许登录?
grep 'nologin$' /etc/passwd | wc -l
5.显示/根目录下的文名件名列表,然后统计根目录下的文件数量?
ls / 显示根目录下的文件名列表
ls / | wc -l 将ls /命令结果用wc -l进行统计
说明:|是管道符,作用是将左边命令的结果传给右边的命令继续处理。
6.分别统计/dev这一层目录中有多少个普通文件?有多少个目录?有多少个块设备文件?有多少个字符设
备文件?有多少个快捷方式(即软链接、符号链接)?
ls -l /dev | grep ^-
ls -l /dev | grep ^d
ls -l /dev | grep ^b
ls -l /dev | grep ^c
ls -l /dev | grep ^l
7.统计系统现在有多少个进程?统计有多少个用户已经登录到linux系统了?
ps -elf | wc -l
who | wc -l 或 w | wc -l
wc
6/35
wc 统计行数 单词数 字节数
wc /etc/passwd
-l 统计行数
wc -l a.txt
-c 统计字节数
wc -c /etc/passwd
-w 统计单词数
wc -w /etc/passwd
-m 统计字符数
wc -m /etc/passwd
统计/bin目录下有多少个命令
ls /bin | wc -l
1690
uniq
uniq 合并连续重复的行
-u 只显示没有连续重复的行
-c 统计连续重复行的次数
-d 只显示连续重复行一次,哪一行连续重复了就显示哪行
-i 忽略大小写
项目准备:准备如下的a.txt文件及其内容。

cat > a.txt <

jim tom root
aa bb cc
aa bb cc
jim
aa bb cc
tom
tom abc
tom
tom abc
EOF

uniq -c a.txt 统计连续出现的次数

1 jim tom root
2 aa bb cc
1 jim
1 aa bb cc
1 tom
1 tom abc
1 tom
1 tom abc

sort a.txt|uniq -c 请跟上一条命令的结果对比

7/35

uniq a.txt 对a.txt文件中连续重复的行只显示一次

jim tom root
aa bb cc
jim
aa bb cc
tom
tom abc
tom
tom abc

uniq -u a.txt 不显示连续重复的行

jim tom root
jim
aa bb cc
tom
tom abc
tom
tom abc

uniq -d a.txt 仅显示连续出现的内容一次

aa bb cc
显示连续重复、未重复的内容,都只显示一次
sort a.txt | uniq
sort a.txt | uniq -c 同上,并统计次数,结果如下
3 aa bb cc
1 jim
1 jim tom root
2 tom
2 tom abc
显示没有重复的行
sort a.txt | uniq -u
jim
jim tom root
diff-patch
准备要比较差异的文件:
head -3 /etc/passwd > /tmp/pas3
head -5 /etc/passwd > /tmp/pas5
vimdiff比较两个文件的差异,不同涂红色
cd /tmp 切换到/tmp目录
vimdiff pas3 pas5 说明: 用:qa!强制关闭所有文件并退出vimdiff命令
注:在vimdiff中可以按i键进入vim的编辑模式修改文件内容。按Esc键退出编辑模式,用:wq保存并退出vim
8/35
编辑器。(用ctrl+w放手后再按w 在窗口之间切换)

diff 文件差异的对比
diff 文件1 文件2
diff练习:
cd /tmp
diff pas3 pas5 对比文件差异,分析显示结果
diff pas5 pas3 对比文件差异,分析显示结果
diff pas3 pas5 > 35.patch 制作补丁(增加内容)
diff pas5 pas3 > 53.patch 制作补丁(减少内容)
打补丁:
升级补丁:
降级补丁:
192.168.8.210
192.168.8.54
安装patch软件:rpm -ivh /dvd/Packages/patch-2.7.1-8.el7.x86_64.rpm
yum install -y patch
patch 打补丁
用法:patch 原文件 补丁文件
实例:35.patch和53.patch补丁文件是diff命令练习生成的。
cd /tmp
patch pas3 35.patch 给pas3文件打补丁,提示如下
patching file pas3
cat -n pas3 读pas3文件并显示行号,发现文件中多了2行内容
patch pas3 53.patch 给pas3文件打补丁,提示如下
cat pas3 发现文件中少了2行内容
stat
stat命令:查看文件的状态信息(即详细属性),包括文件的路径、大小、访问时间(atime)、修改时间
(mtime)、属性修改时间(ctime)、索引号(inode)等属性。
实例:
stat /etc/passwd
Access (最近访问atime)文件最后一次被访问的时间(cat、head查看文件内容)
Modify (最近更改mtime)文件最后一次被修改内容的时间(用vim修改文件内容)
Change (最近改动ctime)文件最后一次被修改过属性的时间(chmod、chown改文件权限、属性)
9/35

复习:09-29 09:00~09:15
进程管理的命令:
查进程:pstree ps top pidof pgrep
进程状态信息:R运行 S休眠 Z僵尸 T暂停 s包含子进程 +进程组 < 高优先级 n
低优先级
杀进程:kill pkill killall xkill
调优先级:renice nice 谦让nice值范围-20~19,nice值越小优先级(PR)越高,PR默认值是20
linux高级命令:
grep匹配关键字 cut取出字符 sort排序 wc统计单词 uniq去重
diff比较差异 patch打补配 which查文件路径 stat查文件状态


locate
updatedb //手动更新查找数据库
说明:安装新软件,创建的新文件,必须updatedb后,才能用locate命令查到。

locate 搜索
命令格式:locate 关键字
特点:
a.可以模糊匹配
b.快速查找
c.基于数据库
说明:必须安装mlocate软件才能使用locate命令。
实例:locate hosts.allow 查找tcpwraper防火墙白名单文件
locate ifcfg-e 查找网卡配置文件及其相关的文件
locate查找数据库(了解即可)
#cat /etc/updatedb.conf 配置文件内容如下,内容修改后需要用updatedb刷新
PRUNE_BIND_MOUNTS="yes" //是否限制搜索,yes表示不在以下文件系统和目录中搜索文件
PRUNENAMES=".git .bzr .hg .svn" //跳过的文件类型,不同后缀之间用空格隔开。不使用则用#号注释
掉了
PRUNEPATHS="/tmp /var/spool /media /home/.ecryptfs" //要跳过的路径,/media 表示其他硬盘
PRUNEFS="NFS nfs nfs4 rpc_pipefs afs binfmt_misc proc smbfs autofs iso9660 ncpfs coda devpts
ftpfs devfs mfs shfs sysfs cifs lustre_lite tmpfs usbfs udf" //要搜索的文件系统
10/35
tar
tr 替换字符 被替换的字符与替换的字符数必须是相等的
用法:tr 旧字符 新字符

cat > a.txt <

root:1:0:bin
bin:2:200:bash
jim:10:300:bash
tom:102:10:bash
bin:2:200:bash
EOF

cat a.txt | tr 20 89 将cat a.txt结果中的20替换成89。

root:1:9:bin
bin:8:899:bash
jim:19:399:bash
tom:198:19:bash
bin:8:899:bash

cat a.txt | tr "a-z" "A-Z" 在替换的时候是一一对应的替换

ROOT:1:0:BIN
BIN:2:200:BASH
JIM:10:300:BASH
TOM:102:10:BASH
BIN:2:200:BASH

cat a.txt | tr 0-4 5-9 将0-4替换成5-9

root:6:5:bin
bin:7:755:bash
jim:65:855:bash
tom:657:65:bash
bin:7:755:bash
\n 换行符

cat a.txt | tr "\n" "+" 将换行符(\n)换成+号

root:1:0:bin+bin:2:200:bash+jim:10:300:bash+tom:102:10:bash+bin:2:200:bash+

cat a.txt | tr -s "o0" 把连续重复的字符替换成一个字符

rot:1:0:bin
bin:2:20:bash
jim:10:30:bash
tom:102:10:bash
bin:2:20:bash

cat a.txt | tr -s "\n" 去掉空白行


练习:
1.只查看你的电脑的ip地址 192.168.1.250
分析思路:1.用什么命令查电脑的ip:ifconfig
2.如何提取ip信息中的ip地址:综合应用grep、head、tail
3.如何将信息中的多个空格变成一个空格:tr -s " "
4.如何用cut命令取出需要的字符: cut -d " " -f 3
11/35
方法一:
说明:以下引号中的紫色是空格,head -2是读前2行,tail -1是读最后一行。

ifconfig | head -2 | tail -1 | tr -s " " | cut -d " " -f 3

192.168.1.250
方法二:以下grep broadcast是匹配关键字broadcast。

ifconfig | grep broadcast | head -1 | tr -s " " | cut -d " " -f 3

192.168.1.252
2.查看/根目录的使用率是多少? 已用的%14
方法一:df -h / | tail -1 | tr -s " " | cut -d " " -f 5
解释:df是显示磁盘空间状态信息,-h是带上k、G等单位
方法二:df -h | head -2 | tail -1 | tr -s " " | cut -d " " -f 5
14%
3.将free -m结果中的mem内存的total、used、free的值取出来
free -m
free -m | head -2 |tail -1 |tr -s " "|cut -d " " -f 2-4
4.将df -h结果中的第1、4、5、6列取出来。
df -h | tr -s " " | cut -d " " -f 1,4-6
5.将lsblk结果中的第1、4、7列取出来。
lsblk | tr -s " " | cut -d " " -f 1,4,7

tar包管理:
作用:实现将文件或目录进行打包压缩、解包解压缩。
windows的压缩软件:WinRaR、好压haozip、360压缩等
为何要压缩?
1、节省空间
2、方便在网络上传输
打包压缩:tar -czvf 包名.tar.gz 要打包的文件
例:tar -czvf etc.tar.gz /etc
du -hs /etc etc.tar.gz 统计文件大小(disk use)
解释:
tar后面的-c是打包(create),z是采用gzip压缩,v显示过程,f是指定包文件名。
du后的-h是显示容量单位,-s是sum统计总大小(通常用于目录)
解包解压缩:tar -xvf 包名.tar.gz -C 解压到的目录
例:tar -xvf etc.tar.gz -C /opt/
ls /opt 显示/opt目录中的文件名列表
rm -rfv /opt/etc 强制删除/opt目录下的etc目录
解释:tar后面的-x是解包解压。
rm是删除文件,-r是递归删除目录,-f强制(即不提示yes/no),-v显示过程。
12/35
sed
sed流编辑器
是一种文本编辑器,默认情况下是不会修改原文件的
也是一种非交互式的编辑器
sed的工作原理
一行一行处理的
当从文件种读取一行后,首先放到模式空间中对该行进行相应的处理,处理完将结果输出到屏幕
上。然后继续处理下一行内容,直到所有行都读取完毕,sed结束。

sed的语法
sed [option] '地址命令' 文件名
选项
-n 静默输出,关闭模式空间的输出,不会输出未匹配到的行
-e 允许进行多项编辑,也就是说对同一行做多次处理,也可以做多点编辑, -e " " -e " " == -e "动作
1" -e "动作2" == '动作1;动作2'
-f 后接sed脚本,指定执行的sed脚本
-r 允许使用扩展正则
-i 直接作用与原文件
练习准备:将/etc/passwd文件的全文内容添加行号后覆盖写入到/tmp/pass文件中。然后查看/tmp/pass
全文内容。
cat -n /etc/passwd > /tmp/pass
cat /tmp/pass

地址定址
地址用于决定对哪些行进行编辑。地址形式可以是数字、正则表达式或二者结合。如果没有指定地址,
13/35
sed将处理输入文件中的所有行。

#:仅处理指定的第#行,以下的1p是指print第1行。

sed -n '1p' /etc/passwd

root:x:0:0:root:/root:/bin/bash

cat -n /etc/passwd |sed -n '5p'

5 lp:x:4:7:lp:/var/spool/lpd:/sbin/nologin

#1,#2:从第#1行开始,到第#2行

cat -n /etc/passwd |sed -n '7,9p'

7 shutdown:x:6:0:shutdown:/sbin:/sbin/shutdown
8 halt:x:7:0:halt:/sbin:/sbin/halt
9 mail:x:8:12:mail:/var/spool/mail:/sbin/nologin
#1,+n:从第#1行开始,再加上n行为止 n表示数字

cat -n /etc/passwd |sed -n '5,+3p'

5 lp:x:4:7:lp:/var/spool/lpd:/sbin/nologin
6 sync:x:5:0:sync:/sbin:/bin/sync
7 shutdown:x:6:0:shutdown:/sbin:/sbin/shutdown
8 halt:x:7:0:halt:/sbin:/sbin/halt
#1~n:从第1行开始,以 n为步长值递增,n表示数字

cat -n /etc/passwd |sed -n '1~2p' 显示文件中从第1行开始的所有奇数行

1 root:x:0:0:root:/root:/bin/bash
3 daemon:x:2:2:daemon:/sbin:/sbin/nologin
5 lp:x:4:7:lp:/var/spool/lpd:/sbin/nologin
7 shutdown:x:6:0:shutdown:/sbin:/sbin/shutdown
9 mail:x:8:12:mail:/var/spool/mail:/sbin/nologin
/keyword/:仅处理符合模式(即匹配关键词)的行,-r是启用正则表达式(RE)

cat /etc/passwd |sed -r -n '/^root/p' 显示以root开头的内容

root:x:0:0:root:/root:/bin/bash

sed -n '/^root/p' /etc/passwd

root:x:0:0:root:/root:/bin/bash
/PAT1/,/PAT2/:从第一次符合PAT1的行到第一次符合PAT2的行,文件如果没有处理玩,下面还有符
合模式的继续处理

sed -n '/^r/,/^s/p' /etc/passwd

sed -n '/^root/,/^s/p' /etc/passwd

sed -n '/adm/,/nobody/p' /etc/passwd

学习准备:执行如下命令,实现创建一个/tmp/pass文件,并用cat查看此文件的全文内容。
head /etc/passwd | cat -n > /tmp/pass
cat /tmp/pass

sed的子命令
sed的子命令告诉sed对指定行进行何种操作,包括打印p、删除d、修改(追加a、插入i、替换s、字符替
换c)。注:黄底色为sed常用的子命令。
p:print,显示地址定位的行,打印

d:delete,删除

sed '/^root/d' /etc/passwd

sed '/^root/d' /etc/passwd >> /tmp/passwd

14/35

i\text:在符合模式的行的上面插入(insert)\以后的text文本,结果将在终端上显示,默认不会更改
原文件(在地址地位行的上一行插入文本)

sed '/^root/i\hello' /etc/passwd

sed '1i\world' /etc/passwd

a\text:在符合模式的行的下面追加(append)\以后的text文本,结果将在终端上显示,默认不会更
改原文件(在地址地位行的下一行插入文本)

sed '1a\world' /etc/passwd

sed '1a\world\

hello\
jim' /etc/passwd
c\text:用新文本修改(change)符合模式的行中的文本(即覆盖掉符合模式行)

sed '1c\world' /etc/passwd

!:对所选行以外的所有行应用命令

sed -n '1,20!p' /etc/passwd

s:用一个字符串替换另一个

sed 's/root/ROOT/g' /etc/passwd //指定替换所有匹配到的

sed 's/root/ROOT/2' /etc/passwd //指定替换第2次匹配到的

sed 's/root/ROOT/' /etc/passwd //指定替换每行第一个匹配到的

sed 's/\//#/' /etc/passwd //把每行第一个匹配到的/替换为

sed 's@/@#@' /etc/passwd

s/// s### s@@@

r:从文件中读 将其他文本内容读到本文本的指定位置
[root@client ~]# cat /tmp/a.txt
aaa
bbb
ccc
ddd
[root@client ~]# cat /tmp/b.txt 123
456
789
564
[root@client ~]# sed '/aaa/r /tmp/b.txt' /tmp/a.txt 将/tmp/b.txt文件内容读到/tmp/a.txt文
件中aaa行的下面
aaa
123
456
789
564
bbb
ccc
ddd

w:将行写入文件
[root@client ~]# cat a.txt
123
456
789
[root@client ~]# cat b.txt
15/35
aaa
bbb
ccc
[root@client ~]# sed '/aaa/w a.txt' b.txt 将b.txt文件中的aaa写入a.txt文件中
aaa
bbb
ccc
[root@client ~]# cat a.txt
aaa
[root@client ~]# cat b.txt
aaa
bbb
ccc

sed '/^root/w a.txt' /etc/passwd 将/etc/passwd文件中以root开头的行写入a.txt文件中

n:对找到匹配的行的下一行进行处理

sed -n '/^root/{n;p}' /etc/passwd //打印/etc/passwd文件中以root开头行的下一行

bin:x:1:1:bin:/bin:/sbin/nologin
q:结束和退出

sed '5q' /etc/passwd //匹配到第5行就退出

1.从root开头的行,到login结尾的行,将sbin替换为/bin

sed '/^root/,/login$/s/sbin/bin/g' /etc/passwd

2.将selinux的模式该为disabled

sed -i '/^SELINUX/s/disabled/permissive/' /etc/sysconfig/selinux

在修改之前先备份文件,备份文件名是在原名称后加.bak

sed -i.bak '/^SELINUX/s/permissive/disabled/' /etc/sysconfig/selinux

ls /etc/sysconfig/selinux*

/etc/sysconfig/selinux /etc/sysconfig/selinux.bak
3.将ip地址替换为192.168.2.230

sed '/^IPADDR/s/192.168.1.251/192.168.2.230/' /etc/sysconfig/network-scripts/ifcfg-eth0

4.用sed分别实现head-1和tail-1的功能

sed -n '1p' /etc/passwd

sed -n '$p' /etc/passwd

5.用sed实现同时打印第一行和最后一行

sed -n -e '1p' -e '$p' /etc/passwd

root:x:0:0:root:/root:/bin/bash
bash:x:506:506::/home/bash:/sbin/nologin

sed -n -e '1p;$p' /etc/passwd

root:x:0:0:root:/root:/bin/bash
bash:x:506:506::/home/bash:/sbin/nologin
6.将root换成root1

sed 's/root/root1/g' /etc/passwd

sed -r 's/(root)/\11/g' /etc/passwd

16/35
选项-f的用法
写sed的脚本 脚本的内容sed的命令

cat sed.test

s/root/ROOT/g
/ftp/i\hello
/lp/a\world123

sed -f sed.test /etc/passwd

1.删除每行的第一个字符

sed 's/^.//' a.txt

2.删除每行的第2个字符

sed 's/^(.)(.)/\1/' a.txt

sed -r 's/^(.)(.)/\1/' a.txt

3.删除每行的倒数第一个字符

sed -r 's/.$//' a.txt

4.删除每行的倒数第2个字符

sed -r 's/(.)(.)$/\2/' a.txt

5.将每行的第一个字符和第二个字符交换

sed -r 's/(.)(.)/\2\1/' a.txt

6.将vsftpd.conf文件中以#开头的行其后没有空格的行显示出来

sed -nr '/^#[^[:space:]]/p' /etc/vsftpd/vsftpd.conf

7.删除/etc/passwd文件中所有的数字

sed -r 's/[0-9]//g' /etc/passwd

8.删除第8行和第20行

sed -e '8d' -e '20d' /etc/passwd

sed '8d;20d' /etc/passwd

9.匹配合法邮箱有[email protected]/>[email protected]
@163.combr/>@com
250br/>[email protected]
[email protected]
[[:alnum:]]+@[[:alnum:]]+.com

sed -nr '/[[:alnum:]]+@[[:alnum:]]+.com/p' a.txt

[email protected]/>[email protected]
[email protected]

17/35
sed练习
学习准备:执行如下命令,实现创建一个/tmp/pass文件,并用cat查看此文件的全文内容。
head /etc/passwd | cat -n > /tmp/pass
cat /tmp/pass
练习:
1.临时删除/tmp/pass文件的偶数行内容
sed '2~2d' /tmp/pass
2.临时删除/tmp/pass文件的奇数行内容
sed '1~2d' /tmp/pass
3.临时删除/tmp/pass文件的3~7行内容
sed '3,7d' /tmp/pass 或 sed '3,+4d' /tmp/pass
4.在/tmp/pass文件的第3行上方insert插入welcome sed cmd内容。在第5行下方append追加bye sed
cmd。
方法一、分两次处理:
sed '3i\welcome sed cmd' /tmp/pass 或 sed '2a\welcome sed cmd' /tmp/pass
sed '5a\bye sed cmd' /tmp/pass 或 sed '6i\bye sed cmd' /tmp/pass
方法二、一次同时处理号:
sed -e '3i\welcome sed cmd' -e '5a\bye sed cmd' /tmp/pass 或如下命令
sed -e '2a\welcome sed cmd' -e '6i\bye sed cmd' /tmp/pass
5.将/tmp/pass文件的第2行整行内容change改成hello sed。
sed '2c\hello sed' /tmp/pass
6.!对所选行以外的所有行应用命令,如下是显示1~7行以外的所有内容。
sed -n '1,7!p' /tmp/pass
7.s替换,用一个字符串替换另一个
sed 's/root/ROOT/g' /tmp/pass //指定替换所有匹配到的
sed 's/root/ROOT/2' /tmp/pass //指定替换第2次匹配到的
sed 's/root/ROOT/' /tmp/pass //指定替换每行第一个匹配到的
sed 's/\//#/' /tmp/pass //把每行第一个匹配到的/替换为#
sed 's@/@#@' /tmp/pass (同上)
sed '1,3s@/@#@g' /tmp/pass //将1~3行中所有的/斜杠符号替换成 # 号
命令格式:地址定址s/old/new/ s#old#new# s@old@new@br/>说明:s和old、new关键字之间的分隔符号可以用/、#、@这3种符号。
8.r读,匹配/tmp/pass的1~3内容,每匹配到一行内容就读取一次/etc/hosts文件。(不常用)
sed '1,3r /etc/hosts' /tmp/pass
9.w写,将匹配到的1~3行内容写入到/tmp/ak文件。(不常用)
sed '1,3w /tmp/ak' /tmp/pass
cat /tmp/ak 读全文
18/35
10.编写一个名称为/sh/sed.sh的脚本。要求如下:
a.带行号显示/tmp/ak.txt全文内容。
b.提示用户输入一段文字,需要将这行文字追加到/tmp/ak.txt文件的最后一行。
c.再执行一遍步骤a的操作。
d.提示用户输入要删除内容的行号,实现用sed -i来删除这行内容。
e.再次执行一遍步骤a的操作。

第1版的脚本,实现基本功能。
vim /sh/sed.sh 脚本内容如下:
#!/bin/bash
while :
do
[ -f /tmp/ak.txt ] && echo '/tmp/ak.txt exsits.' || touch /tmp/ak.txt
cat -n /tmp/ak.txt
read -p 'input you text:' TEXT
echo "$TEXT" >> /tmp/ak.txt
cat -n /tmp/ak.txt
read -p 'input line number or x:' num
if [ $num == x ];then
break
else
sed -i "${num}d" /tmp/ak.txt
cat -n /tmp/ak.txt
fi
done

第2版的脚本:做了一次优化:添加了while循环控制功能。
vim /sh/sed.sh 脚本内容如下:
#!/bin/bash
while :
do
clear
[ -f /tmp/ak.txt ] && echo '/tmp/ak.txt exsits.' || touch /tmp/ak.txt
cat -n /tmp/ak.txt
read -p 'input you text or x or DEL:' TEXT
if [ $TEXT == x -o $TEXT == X ];then
break
elif [ $TEXT == DEL ];then
read -p 'input line number:' num
sed -i "${num}d" /tmp/ak.txt
cat -n /tmp/ak.txt
else
echo "$TEXT" >> /tmp/ak.txt
cat -n /tmp/ak.txt
fi

19/35
AWK
whatis awk :结果是'pattern scanning and processing language'(模式扫描与处理语言)
awk 是一种编程语言,文本编辑器
功能:对文本数据进行汇总和处理,对数据进行排版
注意:awk只能对读入到内存中的信息进行处理,无法对内容所在的源文件内容进行修改。要想修改源文
件内容只能用sed命令来改。
常见内置变量代号:
1.行工作处理模式,读入一行文件,存在“$0”里
2.使用内置变量FS(字段分隔符 )分割一行,存入到$1-$100
3.使用内置变量OFS(输出字段分割符),输出该行
练习准备:执行如下操作。
head /etc/passwd | cat -n > /tmp/pass
cat /tmp/pass
awk语法:
语法一:awk [选项] '/模式匹配/{处理动作1 , 处理动作2......}' 文件列表
语法二:awk [选项] 'BEGIN{动作}{处理动作1 , 处理动作2......}END{动作}' 文件列表
说明:BEGIN是指在读awk指定的文件之前执行的操作,END是指在awk读完指定的文件之后执行的操作。
示例:显示/tmp/pass文件中以 :冒号分隔的第1列内容
[root@jin ~]# awk -F: '{print $1}' /tmp/pass
[root@jin ~]# awk '{print $1}' /tmp/pass
要求:对比以上两条命令结果的区别。
例:执行如下命令,熟悉awk的BEGIN、END语句的用法。
cat /tmp/pass
awk 'BEGIN{print "welcome to awk language."}'
awk 'BEGIN{print "Hosts File Content:"}{print}END{print "File Read Over"}' /tmp/pass
-F 指定字段分割符,awk中默认以 '空格、空白' 作为分隔符
分隔符可以是字母、数字、符号和空格
默认是空格
awk -F" " '{print $1}' passwd
-----单个符号做分隔符
-----多个符号做分隔符
awk -F[:/] '{print $1,$2,$3,$4,$5,$6,$7,$8,$9}' /tmp/pass
1 root x 0 0 root root bin
原文内容 1 root:x:0:0:root:/root:/bin/bash 9个分隔符,10列内容
9个刻度,10段线
awk -F: '/root/ {print $1,$3}' /tmp/pass
1 root 0
10 operator 11
20/35
打印与文件内容无关的
awk '{print "aaa"}' /tmp/pass 根据/tmp/pass内容的行数来循环执行print命令
\n换行符
awk -F: '{print $1"\n"$2}' /tmp/pass
\t制表符
awk -F: '{print $1"\t"$2}' /tmp/pass
打印一句话:root用户的uid是0
awk -F: '/^root/{print $1"的uid是"$3}' /etc/passwd
awk -F: 'BEGIN{OFS=","}/^root/{print $1"的uid是"$3}' /etc/passwd
print 要点:
a 各个输出字段之间用逗号做分隔,而输出时默认是以空格做分隔
b print后面如果不指定字段,那么就会打印一整行
c print输出时默认是有换行符的
例:执行如下2条命令,对比显示结果有何不同。
awk -F: '{print $1,$2,$7}' /tmp/pass 说明: -F:等同于FS=":"这种写法
awk -F: 'OFS=","{print $1,$2,$7}' /tmp/pass 用awk的内置变量OFS指定print输出的分隔符为",逗
号"
awk 'FS=":",OFS=","{print $1,$2,$7}' /tmp/pass 说明:FS=":"等同于 -F:这种写法
练习:
在/tmp/pass文件的每一行后面都打印一个空行
awk '{print $0"\n"}' /tmp/pass
取出网卡的ip地址
centos 6的命令:ifconfig |awk -F" " '{print $2}' |grep addr |head -1 |awk -F: '{print $2}'
centos 6的命令:ifconfig |awk -F [ :] '/addr/{print $2}'
centos 7的命令:ifconfig | awk '/broadcast/{print $2}'
用awk取出ifconfig结果中网卡的MAC地址:
ifconfig | awk '/ether/{print $2}'
用sed、tr、cut命令提取ifconfig结果中的IP地址:
ifconfig | sed -n '/broadcast/p' | tr -s " " | cut -d " " -f 3
说明:tr -s " " 是将多个空格转换成一个空格。
printf 可以格式化(format)输出,默认不换行
使用格式
printf format item1
format的指示符都是以%开头的,后面跟一个字符
%s:表示字符串
%d:十进制整数
%f:表示浮点数(float),也就是小数
%x:表示十六进制数
%o:表示八进制数
修饰符:%Ns N表示显示宽度(是一个数字)
-:表示左对齐,默认是右对齐 %-5s
21/35
对于浮点数:
%4.2f 其中4表示总显示宽度,2表示小数的位数
awk -F: '{printf "%-5s%d\n",$1,$3}' /tmp/pass
格式化输出passwd文件中的用户名,uid,gid三列
awk -F: '{printf "%-15s %-5d %-5d\n",$1,$3,$4}' /tmp/pass
用以下测试四、五的命令格式化输出/etc/passwd文件中第1列(用户名)、7列(用户登录的shell)的内容。
测试一:awk -F: '{print $1,$NF}' /etc/passwd
测试二:awk -F: 'BEGIN{OFS=":"}{print $1,$NF}' /etc/passwd
测试三:awk -F: 'BEGIN{OFS=":";PRINT "username:home_dir"}{print $1,$NF}' /etc/passwd
测试四:awk -F: 'BEGIN{printf "%-12s%s\n","username","home_dir"}/root/{printf "%-12s%s\n",$1,
$NF}' /etc/passwd
测试五:awk -F: 'BEGIN{printf "%-12s%-2s%s\n","username",":","home_dir"}/root/{printf "%-12s
%-2s%s\n",$1,":",$NF}' /etc/passwd
例:执行如下命令,对比两条命令结果的区别。
echo -e 'hello\nwelcome\nlinux\n1234567890' | awk '{printf "%10s\n",$1}'
echo -e 'hello\nwelcome\nlinux\n1234567890' | awk '{printf "%8s\n",$1}'
printf要点:
a printf需要指定格式(format)
b 格式(format)是用来指定后面每个item(条、项目)的输出格式的
c printf默认不会自动打印换行符,需要自己动手添加"\n"
d printf默认没有输出字段分隔符
比较表达式
比较表达式采用关系运算符,用于比较数字与字符串
关系运算符
数字与字符串使用的运算符:
运算符 含义 示例
< 小于
<= 小于或等于
= 赋值
== 相等
!= 不相等

大于
= 大于或等于
#打印/etc/passwd文件中uid等于0的账号信息
awk -F: '$3==0' /etc/passwd 问:为何不写print命令? 答:因为awk默认会执行print命令
#打印出uid大于999的行 passwd
awk -F: '$3>999' /etc/passwd
以冒号为分隔打印第七列是/bin/bash的行
awk -F: '$7=="/bin/bash" ' /etc/ passwd 比较字符串时需要用双引号将字符串引起来
条件表达式:
awk [选项] '{if(条件表达式) {结果1} else{结果2}}' 最外面的大括号用于行处理
22/35
对/etc/passwd文件所有用户做打印输出,要求当$3(即第3列uid)等于0的时候,打印$1(即第1列
username)是管理员账户,否则打印$1(即第1列username)是普通用户
测试一:awk -F: '{if($3==0) {print $1,"is admin"} else{print $1,"is user"}}' /etc/passwd
测试二:awk -F: '{if($3==0) {print $1,"是管理员账户"} else{print $1,"是普通用户"}}' /etc/passwd
算术运算: +,-,*,/,乘方
#打印uid10的结果大于500的用户信息。
awk -F: '$3
10 > 500' /etc/passwd
#用awk计算2的7次方的值。
awk 'BEGIN{print 2^7}
awk 'BEGIN{z=2^7;print z}' 说明:awk中调用变量的值跟c语言是相同的,直接用变量名,无需$符

shell脚本中的数学运算方法:(技术回顾)
方法 实例
expr expr 2 + 5
let i=1;let i++;echo $i
$(()) echo $((1+23))
$[] echo $[1+2
3]
echo $[2
7] 输出2的7次方的值
逻辑操作符和复合运算
&& 逻辑与 a&&b
|| 逻辑或 a||b
! 逻辑非 !a
awk -F: '$3 >50 && $3<80' /etc/passwd 找出$3大于50且小于80的行,即uid的值是50~80之间的信

找出$3大于100且$4小于120的行
找出$3小于150或$4小于100的行
找出不再第一题范围内的行
awk -F: '$3>100&&$4<120' passwd
awk -F: '$3<150||$4<100' passwd
awk -F: '!($3>100&&$4<120)' passwd
awk的模式匹配(文本匹配)
/key/ 匹配key关键字
x ~ /key/ 匹配正则 其中key可以是正则表达式,x通常是$1~$n
awk -F: '$1 ~/ro/{print $0}' /etc/passwd 打印/etc/passwd中第1列包含ro字符的内容
x !~/key/ 不匹配正则
awk -F: '$7 !~/nologin$/' /etc/passwd 打印/etc/passwd中第7列(即登录shell)除了nologin结尾的用
户信息
常见的模式
首先,准备工作
cd /etc
然后,执行如下命令
1.空模式
awk -F: '{print $1}' passwd
23/35
2.固定模式地址
awk -F: '$1 ~/^ro/{print $1}' passwd
3.范围模式地址
awk -F: '/^ro/,/^lp/{print $1}' passwd
注意:awk中不支持行号定址
反例:
awk -F: '2,5{print $1}' passwd
推翻以上蓝底色的注意说法:
用awk的内置变量NR(即记录号Number Record)变量就可以实现类似行号定址的功能
awk 'NR==2,NR==5' /etc/passwd 输出记录号NR为2~5的信息
awk的一部分小结:
表达式
$3>100
不常见模式
语法:awk 'BEGIN{}'
实例:
awk 'BEGIN{print "welcome to awk language."}'
awk -F: 'BEGIN{OFS="@"} {print $1,$2}' /etc/passwd
awk 'BEGIN{FS=":";print "user"} {print $1}' /etc/passwd
语法:awk 'END{}'
实例:awk '{print $0} END{print "passwd"}' passwd
交换passwd文件中的第一列和第七列并且打印出来(以冒号为分隔符)
awk -F: '{print $7":"$1}' passwd
awk -F: 'BEGIN{OFS=":"} {print $7,$1}' passwd
awk 'BEGIN{FS=":";OFS=":"} {print $7,$1}' passwd
打印uid和gid不同的用户的用户名、uid及gid
awk -F: '$3!=$4{print $1,$3,$4}' passwd
内置变量(即awk自带的内定默认变量)
$0 保存当前的整行记录的内容,例 awk -F: '{print $0}' passwd
NR 记录号(Number Record),每处理完一条记录,NR值就加1,记录的是文件的总值
例 awk -F: '{print NR,$0}' passwd 输出passwd文件内容并显示行号,比cat -n的效果好
FNR 处理多个文件时分别独立统计自己的NR值
准备文件:
head -5 /etc/passwd > /tmp/p5
head -3 /etc/passwd > /tmp/p3
例:执行如下命令,找出以下两条awk命令结果中的行号有何不同?
cd /tmp ;
awk -F: '{print NR,$0}' p5 p3
awk -F: '{print FNR,$0}' p5 p3
NF 表示当前行的字段数(Number Field)
awk -F: '{print NF,$0}' passwd group
df | awk '{print NF,$0}'
24/35
FS 输入字段分隔符,默认是空白
awk 'BEGIN{FS=":"} {print $1,$2}' passwd
OFS 输出字段分隔符,默认是空白
awk 'BEGIN{FS=":";OFS="@"} {print $1,$2}' passwd
RS 输入记录分隔符(记录的是行与行之间的分隔),默认是一个换行符
awk 'BEGIN{RS=":"} {print }' passwd
ORS 输出记录分隔符
awk 'BEGIN{ORS=":"} {print }' passwd
打印passwd文件的奇数行的行号和内容
awk 'NR%2==1{print NR,$0}' passwd
打印passwd文件的偶数行的行号和内容
awk 'NR%2==0{print NR,$0}' passwd
打印passwd文件的倒数第三个字段
awk -F: '{print $(NF-2)}' passwd

从第一行开始,每隔2行打印一次
awk 'NR%3==1{print NR,$0}' passwd
自定义变量
命名:由字母,下划线,数字组成,不能以数字开头,最好不要使用关键字(即不要使用想if、else等命令
关键字作为变量名)
变量名=变量的值
变量名="字符串" 一定要有双引号
awk 'BEGIN{a=2;print a}'
awk 'BEGIN{a="asd";print a}'
赋值运算
a++ 等同于 a=a+1
++a
a+=5 a+=$3
a=a+5 a=a+$3
awk 'BEGIN{a=2;a++;print a}'
awk 'BEGIN{a=5;a--;print a}'
awk 'BEGIN{a=1;a+=5;print a}'
awk -F: '{a+=$3;print a}' /etc/passwd 对/etc/passwd中的uid做累加计算
awk流程控制
条件判断
if语句
{if(表达式) {语句,语句....}}
#显示/etc/passwd中gid大于999的信息
awk -F: '{if($4>999){print $0}}' /etc/passwd
25/35
#统计系统用户的数量(cat /etc/login.defs)
awk -F: '{if($3>201&&$3<999){i++;print $0}} END{print i}' /etc/passwd
#(awk统计中必须掌握的结构)统计/etc/passwd文件中登录shell时/bin/bash有多少个,不是/bin/bash的有
多少个?
awk -F: '{if($NF=="/bin/bash") {i++} else{j++}} END{print "不是/bin/bash的有:"j "\n是/bin/bash的
有:"i}' /etc/passwd
解释:NF是awk的内置变量,表示当前行的字段数(即列数),$NF表示调用NF变量的值
注意:awk调用内置变量值的时候需要用$变量名,而调用自定义变量值的时候不需要$符号。
循环
while:
打印1-10
awk -F: 'BEGIN{i=1;while(i<=10){print i;i++}}' passwd
awk -F: '{i=1;while(i<=10){print i;i++}}' passwd
for
将每一行打印10次
awk -F: '{for(i=1;i<=10;i++) {print $0}}' passwd
数组
awk -F: '{username[i++]=$1} END{print username[1]}' passwd
统计passwd文件中各种shell类型
awk -F: '{shell[$NF]++} END{for(i in shell){print i,shell[i]}}' passwd
shell[$NF]++定义了一个名称为shell的数组,$NF在passwd中是shell名称,for(i in shell) print i,shell[i],用
for循环取出数据来,这时候i取的是下标,shell[i]是里面存储的数据,也是$NF出现的次数
统计netstat -ant最后一列有多少个
netstat -ant |grep tcp |awk -F" " '{state[$NF]++}END{for(i in state){print i,state[i]}}'
统计当前用户名为4个字符的用户
$1 ~/^....$/
awk内置变量
内置变量
$0 保存当前的整行记录的内容
awk -F: '{print $0}' passwd
NR 记录号,每处理完一条记录,NR值就加1,记录的是文件的总值
awk -F: '{print NR,$0}' passwd
FNR 处理多个文件时分别统计自己的NR值
awk -F: '{print FNR,$0}' passwd /etc/shadow
NF 表示当前行的字段数(Number Field,即列数)
awk -F: '{print NF,$0}' passwd /etc/shadow
FS 输入字段分隔符,默认是空白
awk 'BEGIN{FS=":"} {print $1,$2}' passwd
OFS 输出字段分隔符,默认是空白
awk 'BEGIN{FS=":";OFS="@"} {print $1,$2}' passwd
RS 输入记录分隔符(记录的是行与行之间的分隔),默认是一个换行符
awk 'BEGIN{RS=":"} {print }' passwd
ORS 输出记录分隔符
26/35
awk 'BEGIN{ORS=":"} {print }' passwd
awk逻辑控制
awk流程控制
条件判断
if语句
{if(表达式) {语句,语句....}}
#显示/etc/passwd中gid大于999的信息
awk -F: '{if($4>999){print $0}}' /etc/passwd
#统计系统用户的数量(cat /etc/login.defs)
awk -F: '{if($3>201&&$3<999){i++;print $0}} END{print i}' /etc/passwd
#(awk统计中必须掌握的结构)统计/etc/passwd文件中登录shell时/bin/bash有多少个,不是/bin/bash的有
多少个?
awk -F: '{if($NF=="/bin/bash") {i++} else{j++}} END{print "不是/bin/bash的有:"j "\n是/bin/bash的
有:"i}' /etc/passwd
解释:NF是awk的内置变量,表示当前行的字段数(即列数),$NF表示调用NF变量的值
注意:awk调用内置变量值的时候需要用$变量名,而调用自定义变量值的时候不需要$符号。
思考:
1.统计/etc/passwd文件中uid大于999的用户数量有多少个?
awk -F: '{if($3<10){i++}}END{print "UID大于999的用户总数:" i "个"}' /etc/passwd
2.统计/etc/group文件中gid小于10的组有多少个?
awk -F: '{if($3<10){i++}}END{print "GID小于10的组总数:" i "个"}' /etc/group
3.用一条awk命令统计ls -l /dev结果中目录的总数、文件(包括普通文件、块设备、字符设备、套接字、
管道文件、软链接文件)的总数。
ls -l /dev | awk '{if($1 ~ /^d/){i++}else{j++}}END{print "当前目录中目录的 数:" i "\n文件的总数:" j}'
4.用awk统计apache默认网站主页的访问量,要求统计的是状态码为200(即访问成功)的访问量是多少?
[提示,apache默认网站的访问日志位置是/etc/httpd/logs/access_log]
yum install -y httpd curl
systemctl restart httpd
echo 'Thist is apache test site.' > /var/www/html/index.html
curl 127.0.0.1
curl 192.168.11.11
grep 'access_log' /etc/httpd/conf/httpd.conf
cat /etc/httpd/logs/access_log 读apache默认网站的访问日志文
awk '{if($9=="200"){i++}}END{print "200状态码的访问量是:" i }' /etc/httpd/logs/access_log
循环语句
while:
打印1-10数字
awk 'BEGIN{i=1;while(i<=10){print i;i++}}'
打印10-1数字
awk 'BEGIN{i=10;while(i>=1){print i;i--}}'
27/35
awk -F: '{i=1;while(i<=10){print i;i++}}' /etc/passwd
for
将每一行打印10次
awk -F: '{for(i=1;i<=10;i++) {print $0}}' /etc/passwd
数组
awk -F: '{username[i++]=$1} END{print username[1]}' passwd
统计passwd文件中各种shell类型
awk -F: '{shell[$NF]++} END{for(i in shell){print i,shell[i]}}' passwd
shell[$NF]++定义了一个名称为shell的数组,$NF在passwd中是shell名称,for(i in shell) print i,shell[i],用
for循环取出数据来,这时候i取的是下标,shell[i]是里面存储的数据,也是$NF出现的次数
统计netstat -ant最后一列有多少个
netstat -ant |grep tcp |awk -F" " '{state[$NF]++}END{for(i in state){print i,state[i]}}'
统计当前用户名为4个字符的用户
$1 ~/^....$/
awk数组
数组
#用awk打印/etc/passwd每个用户名。
awk -F: '{username[i++]=$1} END{print username[1]}' /etc/passwd
解释:统计passwd文件中各种shell类型
awk -F: '{shell[$NF]++} END{for(i in shell){print i,shell[i]}}' /etc/passwd
shell[$NF]++定义了一个名称为shell的数组,$NF(即number field行中的字段数)在passwd中是shell名称,
for(i in shell) print i,shell[i],用for循环取出数据来,这时候i取的是下标,shell[i]是里面存储的数据,也是
$NF出现的次数
统计netstat -ant最后一列有多少个
netstat -ant |grep tcp |awk -F" " '{state[$NF]++}END{for(i in state){print i,state[i]}}'
统计当前用户名为4个字符的用户
$1 ~/^....$/
awk实例
练习:执行如下命令,了解以当天的 年-月-日为文件名创新文件的方法。
date +%F
touch $(date +%F).txt 注:此行命令是先执行$()中的命令,然后执行touch名令。
例:执行如下命令,熟悉功能。
basename /etc/sysconfig/network-scripts 提取路径中的文件名
dirname /etc/sysconfig/network-scripts 提取路径中的路径部分
28/35
练习:编写一个名称为/sh/tar.sh的脚本。要求提示用户输入tar打包压缩的路径,回车确认后判断目录路径
是否存在,如果存在就用tar命令将此目录打包压缩成/opt/目录名-年-月-日.tar.gz文件,否则提示目录不存
在。
#!/bin/bash
read -p '输入压缩路径:' Dir
[ -d $Dir ] && tar -czvf /tmp/basename ${Dir}-echo $(date +%F).tar.gz $Dir || echo '目录不存
在'
脚本编写的实施:
1.根据需求分析需要用的命令
#!/bin/bash
read -p 'input your directory path:' Dir
dir_name=$(basename $Dir)
DT=$(date +%F)
test -d $Dir
if [ $? -eq 0 ];then
tar -czvf /opt/${dir_name}-${DT}.tar.gz $Dir
else
echo "$Dir not exsits."
fi
练习:将df -h /boot 的结果传给awk、sed进行处理,用awk做判断输出处理,如果可用空间小于
900M,就提示'/boot'可用空间小于900M,否则提示可用/boot空间充足。
写法一(正常解题思维):
a=$(df -h /boot | sed 's/M//g' |awk 'NR==2{print $4}')
[ $a -lt 900 ] && echo '/boot 可用空间小于900M' || echo '/boot空间充足'
写法二:
test $(df -h /boot | sed 's/M//g' |awk 'NR==2{print $4}') -lt 900 && echo '/boot 可用空间小于
900M' || echo '/boot空间充足'
写法三:
[ $(df -h /boot | sed 's/M//g' |awk 'NR==2{print $4}') -lt 900 ] && echo '/boot 可用空间小于900M'
|| echo '/boot空间充足'
练习:用ls /dev/sda?显示sda硬盘的分区文件列表,然后输出'sda硬盘有多少个分区。'
方法一:echo "sda 硬盘有 $(ls /dev/sda? | wc -l) 个分区."
方法二:ls /dev/sda? | awk '{if($1 ~ sda){i++}}END{print "sda硬盘有" i "个分区"}'
以下练习中可能综合应用cut、grep、sed、head、tail等命令。
1.提取ifconfig命令结果中的IP地址信息。
方法一:ifconfig | awk '/broad/{print $2}'
x=$(ifconfig | awk '/broad/{print $2}')
echo $x
2.编写一个名称为/sh/mem.sh,要求判断free -h结果中物理内存free空闲空间低于100M,就提示'free的
值、内存不足,否则提示free的值、内存够用。
free -h | awk '/Mem/{print $4}' | sed 's/M//
29/35
FREE=$(free -h | awk '/Mem/{print $4}' | sed 's/M//')
echo $FREE
/sh/mem.sh脚本内容
#!/bin/bash
FREE=$(free -h | awk '/Mem/{print $4}' | sed 's/M//')
[ $FREE -lt 100 ] && echo 'memory little' || echo 'memroy enough'
3.编写一个名称为/sh/df.sh,要求判断df -h 结果中/boot目录空闲空间低于100M,就提示'/boot的free
值、空间不足,否则提示/boot的free值、空间够用。
/sh/df.sh脚本内容
#!/bin/bash
DF=$(df -h | awk '/boot/{print $4}' | sed 's/M//')
[ $DF -lt 100 ] && echo 'storage little' || echo 'storage enough'
read
read:读。从一个通道读。通常用于提示用户输入某个信息,并把输入的信息赋值给一个变量。
用法:read [选项] 变量名
选项:
-p '提示符' :p是print打印的缩写,即在屏幕显示'提示信息'
-t 超时 : 指定等待用户输入信息的时间,超过这个时间会自动跳过此步
-a 数组名 : 将所输入的信息赋值给一个数组
-s 静默执行,即不显示用户输入的信息。通常用于提示输入密码的场合。
变量名1 变量名2 ... : 将输入的信息赋值给不同的变量,信息间默认以空格分隔。
例:#执行如下操作
read -t 3 -p '请在3秒内输入密码:' X
echo $X
#提示用户输入一些文本,将输入的信息赋值给一个名称为arr的数组。
read -a arr -p 'input any text:'
#输出arr数组的值
echo ${arr[*]} 或 echo ${arr[@]} 输出arr数组所有元素的值
echo ${arr[0]},${arr[2]} 输出arr数组的第0、2个元素的值
#提示用户输入信息,并把信息中的每个元素赋值给不同的变量。
read -p '请输入一些数字:' x y z
#输出以上变量x、y、z的值
echo $x,$y,$z
#提示'按回车键继续...'
read -p 'Press Enter to continue...'
脚本 文本文件 解释型语言
30/35
把平时在终端上执行的非交互式命令一次性写到一个文本文件里面,然后给他们指定一个命令解释器
交互式命令
输入命令一次性执行不完
passwd
非交互式命令
输入命令一次性执行
ls /tmp
shell脚本
1.脚本的名字 .sh为结尾 (只要是已.sh结尾的文件系统就认定是脚本文件)
vim a.sh
2.在vim编辑器内编辑内容(也可以用其他编辑器写脚本)
1)指定命令解释器
#! 用来指定命令解释器
#!/bin/bash
2)写注释(注释是不可以被脚本解释的)

注释 除了第一行的#之外 其他以#开头的行全部都是注释行

3)写版本

version:0.0

4)写作成者

autor:tom

5)写要执行的命令(写非交互式的命令)

查看当前系统的信息
1.统计系统的使用率为:

df -h

2.当前登陆的用户为:

echo $USER

3.本机的ip地址:

ifconfig eth0 | head -2 | tail -1 |cut -d : -f 2 | cut -d " " -f 1

4.当前的日期为:

date

5.当前可以登陆到系统的用户有几个:

grep -c 'bin/bash' /etc/passwd

6.当前系统的主机名:

hostname

vim a.sh
#!/bin/bash
#查看当前系统的信息
#version:0.0
#autor:jim
31/35
echo "磁盘的使用率为:"
df -h
echo
echo "当前登陆的用户为:$USER"
echo
echo "本机的ip地址:ifconfig eth0 | head -2 | tail -1 |cut -d : -f 2 | cut -d " " -f 1"
echo
echo "当前的日期为:"
date
echo
echo "当前可以登陆到系统的用户有几个:grep -c "bin/bash" /etc/passwd"
echo
echo "当前系统的主机名为:hostname"
执行脚本

bash a.sh 用/bin/bash命令执行 在子shell中执行

sh a.sh 用/bin/sh命令执行

. a.sh 在当前shell中执行 # source a.sh

./a.sh 在子shell中执行 需要执行权限

/root/a.sh 在子shell中执行 需要执行权限

dialog
dialog
dialog的每一个部件都提供两种格式的输出:
1.将所有输出都输出到标准错误输出里,不显示在屏幕
2.退出状态码
软件包:dialog
yum install dialog -y
部件返回的任何数据都会将数据发送到标准错误输出,可以将标准错误输出到文本文件
dialog --inputbox "请输入你的年龄" 10 20 2> /tmp/err.txt
解释:10是对话框的高度,20是长度。
如果选择yes或ok,返回状态码为0,如果选择cancel或no,返回状态码为1,ESC 为255
dialog --title "please anser" --yesno "is ok ?" 10 20
echo $?
1.消息框
格式:dialog --msgbox “test” height width
dialog --title testing --msgbox "this is test" 10 20
2.yesno框
格式:dialog --yesno “test” height width
32/35
3.输入框
格式:dialog --inputbox “test” height width
dialog --title "passwd" --inputbox "please input your passwd" 10 20 2>aaa
4.密码框
dialog --passwordbox “test”
dialog --title "passwd" --passwordbox "passwd" 10 20 2>aaa
--insecure 使用密文
dialog --title "passwd" --insecure --passwordbox "passwd" 10 20 2>aaa
5.文本框
格式:dialog --testbox file heigth width
dialog --title "this passwd" --textbox /etc/passwd 15 50
6.菜单框
dialog --menu “test” height width menu-height 1 “cat /etc/passwd” 2 “” 3 “”
dialog --title "this menu" --menu "mune" 2 20 4 1 "cat /etc/passwd" 2 "cat /etc/shadow" 3 "cat /etc/
fstab" 2>aaa
7.文件选框
dialog --fselect filepath heigth width
dialog --title "file" --fselect /root/ 10 139
8.复选框(多选框)
dialog --checklist “test” heigth width menu-height 1 “” 2 “”
dialog --backtitle "chechlist" --checklist "test" 10 20 10 memory memory_size 1 Dsik Disk_Size 2
9.显示日历
dialog --calendar “Data” height widht day month year
dialog --title "Calendar" --calendar "Data" 5 15
指定年月日
dialog --title "Calendar" --calendar "Data" 5 15 1 12 2013
10.进度框架
dialog --gauge test height width []
dialog --title "install pro" --gauge "installation" 10 20 10 不能动的进度框
for i in {1..100};do sleep 1;echo $i;done |dialog --title "install pro" --gauge "installation" 10 20 可以
动的进度框
dialog --title "menu" --menu "请选择你要的操作" 10 20 3 1 "注册" 2 "登陆" 3 "退出" 2>dialog
a=cat dialog
if [ $a == 1 ]
then
dialog --title "注册" --inputbox "please input your name" 10 20 2>aaa
fi
zenity
33/35
zenity
必须要有桌面才可以使用,如果要在虚拟机里使用,请直接打开虚拟机桌面
--calendar 显示日历对话框
--text 设定对话框的文字
--day 设定日
--month 设定月
--year 设定年
--date-format=模式 设定返回日期的格式
例子:
zenity --calendar --title="select a data" --text="data" --day=10 --month=5 --year=2014 --date￾format=%y%m%d
常见选项:
zenity --help 查看zenity帮助
--title=“标题” 设定对话框标题
--window-icon=图标路径 设定窗口图标
--width=宽度 设定宽度
--heigth=高度 设定高度
--timeout=超时时间 设定对话框的超时秒数
应用程序的选项:
--calendar 显示日历对话框
--entry 显示文本输入对话框
name=$(zenity --entry --text="请输入你的用 户名" --entry-text="wing")
--error 显示错误对话框
--info 显示信息对话框
--file-selection 显示文件选择框
--list 显示列表对话框
--notification 显示通知
--progress 显示进度标识对话框
--question 显示问题对话框
--warning 显示警告对话框
--scale 显示范围对话框
--text-info 显示文本信息对话框
显示文本输入对话框
用法 : zenity [选项....] zenity --help
[root@today ~]# zenity --help-entry
文字输入选项:
--entry 显示文本输入对话框
--text=“文字” 设定对话框文字
--entry-text=“文字” 设定输入文字
--hide-text 隐藏输入文字
zenity --entry --text="请输入你的用 户名" --entry-text="wing"
34/35
消息框:
zenity可以创建四种消息框
错误 信息 询问 警告
显示错误消息框: --error
zenity --help-error
用法:
zenity [选项...]
错误选项
--error 显示错误对话框
--text=文字 设置对话框文字

zenity --error --text="用户名不存在"

显示信息对话框: --info
用法:
zenity [选项...]
信息选项
--info 显示信息对话框
--text=文字 设置对话框文字

zenity --info --text="你要做什么?"

显示询问对话框:--question
用法:
zenity [选项...]
问题选项
--question 显示问题对话框
--text=文字 设置对话框文字
--ok-label 设定确定按钮
--cancel-label 设定取消按钮

zenity --question --text="nixiangganma "

显示警告对话框:--warning
用法:
zenity [选项...]
问题选项
--warning 显示问题对话框
--text=文字 设置对话框文字

显示文件选择框:--file-selection
用法:
zenity --file-selection --filename="b.sh"
显示列表对话框:--list
用法:
zenity [选项...]
35/35
列表选项
--list 显示列表对话框
--text=文字 设置对话框文字
--column=列 设置列标题
--checklist 第一列使用复选框
--radiolist 第一列使用单选钮
--imagelist 第一列使用图像
--separator=分隔符 设置输出分隔符
--multiple 允许选中多行
--editable 允许更改文字
--print-column=序号 打印指定列(默认为 1。“全部”可用于打印全部列)
--hide-column=序号 隐藏指定列
--hide-header 隐藏列头
zenity --list --text="test" --column="请选 择" --column="文件名称" --column="文件大小" aa a 8 bb b
15 cc c
显示通知:--notification
用法:
zenity [选项...]
通知图标选项
--notification 显示通知
--text=文字 设定通知文字
--listen 在 stdin 上监听命令
--hint=文字 Set the notification hints
zenity --notification --text="你输入的不是2" && zenity --warning --text=“你真2”
显示进度对话框:--progress
用法:
zenity [选项...]
进度选项
--progress 显示进度指示对话框
--text=文字 设置对话框文字
--percentage=百分比 设定初始百分比
--pulsate 跳动进度条
--auto-close 达到 100% 时关闭对话框
--auto-kill 若按下取消按钮则终止父进程
--no-cancel 隐藏取消按钮
注意:pulsate和percentage只能选一个