咱有什么说什么,说实在的,这部分我感觉是liunx中最乱的地方,也是最不好学的,也是能看出人最专业的地方,明白我的意思了吧,,,,自己体会。shell中的数组、正则表达式、sed、awk的使用_表达式




数组 declare

定义变量的类型

declare 选项 变量名=值


选项有

-i ×××的变量

-r 只读的变量 定义就不可以修改了

-x 系统环境变量

-a 数组变量 想把多个变量定义成一个组


---------

[root@xu ~]# declare -i a=1 b=2

[root@xu ~]# declare -i c=$a*$b

[root@xu ~]# echo $c

2


---------

创建数组

[root@xu ~]# declare -a teaname


----------

查看数组

[root@xu ~]# declare -a


------

数组赋值 (添加信息)


数组名 [元素下标]=值


----------

给数组添加内容


[root@xu ~]# teaname[0]=yzs

[root@xu ~]# teaname[1]=xly

[root@xu ~]# teaname[2]=lili

[root@xu ~]# declare -a

declare -a BASH_ARGC='()'

declare -a BASH_ARGV='()'

declare -a BASH_LINENO='()'

declare -a BASH_SOURCE='()'

declare -ar BASH_VERSINFO='([0]="3" [1]="2" [2]="25" [3]="1" [4]="release" [5]="i386-redhat-linux-gnu")'

declare -a DIRSTACK='()'

declare -a FUNCNAME='()'

declare -a GROUPS='()'

declare -a PIPESTATUS='([0]="0")'

declare -a teaname='([0]="yzs" [1]="xly" [2]="lili")'


------------

[root@xu ~]# serip=(1.1.1.1 2.2.2.2 3.3.3.3)

[root@xu ~]# declare -a |grep serip

declare -a serip='([0]="1.1.1.1" [1]="2.2.2.2" [2]="3.3.3.3")'


---------------

输出数组元素

[root@xu ~]# echo ${serip[1]}

2.2.2.2


输出全部元素

[root@xu ~]# echo ${serip[@]}

1.1.1.1 2.2.2.2 3.3.3.3

[root@xu ~]# echo ${serip[*]}

1.1.1.1 2.2.2.2 3.3.3.3


----------

统计数组元素的个数

[root@xu ~]# echo ${#serip[*]}

3


--------

统计模个元素值的长度 (第一个代表0 第二个代表1 1.1.1.1 是7位.也算)

[root@xu ~]# echo ${#serip[0]}

7


---------

输出模个位置的元素 从所有的数组里选取第2个元素

[root@xu ~]# echo ${serip[@]:2}

3.3.3.3


------

[root@xu ~]# hostname=(www.baidu.com www.sina.com www.uplooking.com)

[root@xu ~]# echo ${hostname[2]}

www.uplooking.com

[root@xu ~]# echo ${hostname[2]:0}

www.uplooking.com

[root@xu ~]# echo ${hostname[2]:4}

uplooking.com

[root@xu ~]# echo ${hostname[2]:4:9} 从第几个开始输出 输出几位

uplooking


--------

把从屏幕输入的三次IP 地址 存放到数组serip里

把数组色日跑得三个元素值输出到屏幕上

1 #!/bin/bash

2 read -p "请输入第一个IP地址" a[0]

3 read -p "请输入第二个IP地址" a[1]

4 read -p "请输入第三个IP地址" a[2]

5

6 echo ${a[@]}


用循环的方式给数组赋值


1 #!/bin/bash

2 read -p "清输入监控主机的台数:" a

3 i=0

4 while [ $i -lt $a ]

5 do

6 read -p "清输入第一台主机ip:" serip[$i]

7 let i++

8 done

9

10

11 echo ${serip[@]}




用循环的方式输出数组元素

1 #!/bin/bash

2 read -p "清输入监控主机的台数:" a

3 i=0

4 while [ $i -lt $a ]

5 do

6 read -p "清输入第一台主机ip:" serip[$i]

7 let i++

8 done

9 echo "-------------------------------------------"

10 for i in `echo ${serip[@]}`

11 do

12 echo $i

13 done

14 echo "----------------------------------------"

15 for ((i=0;i<$a;i++))

16 do

17 echo ${serip[$i]}

18 done



-----------------------------------------------------------------------

-----------------------------------------------------------------------

正则表达式

正则 是用符号来描述有相同属性的数据


\转意 在普通情况下 是把特殊符号变成普通符号

在grep (正则表达式中) 是把普通符号变成特殊符号 直接写是没有特殊意义的




grep [选项]‘正则表达式’ 文件名

命令| grep [选项]‘正则表达式’ 文件名


处理方式


逐行处理 把与正则表达式匹配的行输出 以行为处理单位


-------------------

正则符号

^ 开头

$ 结尾

^a 以a为开头

^ 以 为开头的

^# 以#开头的

$ 结尾

^$ 有开头和结尾

a$ 以a为结尾的

^ab$ 以a开头 b为结尾的

. 匹配任意单个字符(任意一位)

\.$ 配置以点结尾的行

* 前面的字符出现0此或者无穷次


* >=o o个到多个

+ >= 1 1 个到多个



\? 是0次或1



***************************************************


用grep 的时候 下面的这几个必须要转意

^ $ . [] * \? \+ \{ \} \< >/ \| \( \)


****************************************************


() 保存 匹配的模式


(ab) 一个整体



^pac*j$ c出现0此或者无穷次

^p(ac)*j$ *前面的ac出现0此或者无穷次



[ ] 匹配范围内的也就是任意一个


[afb] 包括 afb里任意一个都匹配

[a-d]只要里面的有任意一个都匹配

[a-Z]所有的字母


.* 表示所有字符




[^a-c] ^在[]表示取反 没有a到c的都匹配

^[a-c] 以abc开头的

^[^a-c] 不以abc开头的行



| 或

(abc|def)* abc出现o到无穷此 或者 def出现0次或者无穷

w(abc|def)x 表示wabcx 或者是wbefx 都匹配



匹配次数

\{ \}指定前面的表达式出现的次数


g\(oo\)\{1\}d =good

g\(oo\)\{2\}d =gooood 俩个oo 出现2次

{2,} >=2

{2,4} >=2 and <=4 最小出现2次最大出现4次




\< \<shc 匹配行里以shc为首的单词


>/ a>/ 匹配行里以a为结尾的



正则选项

-v 取反

-i 不分大小写

-n 显示匹配文件的行号

-c 显示正则表达式匹配的行数

-q 不显示输出的信息


去掉前面有空格 和#的 只要有效行

[root@xu opt]# grep -v "^$\|^#" /etc/httpd/conf/httpd.conf



面试的时候肯能出现的匹配


匹配出make地址HWADDR

[root@xu opt]# grep "\([0-9a-fA-F]\{2\}:\)\{5\}[0-9a-fA-F]\{2\}" /etc/sysconfig/network-scripts/ifcfg-eth0

HWADDR=54:E6:FC:71:17:45



ip地址



域名 www . baidu . com

[root@xu opt]# grep "[a-z0-9]\{1,\}\.[a-z0-9]\{1,\}\.[a-z]\{2,3\}" ifcfg-eth0 aaawww.baidu.com




主机名


url 地址 论坛上贴网址的



邮箱



---------------------------------------------------

sed 交换式的流编辑器

打印 删除 替换 复制 剪切 粘贴 导入 导出


处理的时候以行为单位 对文件进行逐行处理




用法

sed [ 选项] “动作” 文件名字

命令 | sed [ 选项] “动作” 文件名字



打印 “p”

会显示两行 一个是原数据 一个是处理完以后的数据


sed "p" /etc/passwd 有两行

sed "p;p" /etc/passwd 多次动作用;间隔


sed "p;=" /etc/passwd = 显示行号


root:x:0:0:root:/root:/bin/bash p打印的内容

1 =显示的内容

root:x:0:0:root:/root:/bin/bash 原数据

bin:x:1:1:bin:/bin:/sbin/nologin

2

bin:x:1:1:bin:/bin:/sbin/nologin



-n 不显示原数据 只显示输出后的结果

[root@xu opt]# sed -n "p" /etc/passwd


打印模几行

[root@xu opt]# sed -n "2,5p" /etc/passwd

[root@xu opt]# sed -n "1p;3p;5p" /etc/passwd



---------------------

删除 d


只删除第一行的显示内容

[root@xu opt]# sed "1d" /etc/passwd


[root@xu opt]# sed "1d;3d" /etc/passwd

起始行 结果行

[root@xu opt]# sed "1,3d" /etc/passwd



加 -i 选项 是操作原文件(慎用)

[root@xu opt]# sed -i "1d" /etc/passwd



------------------------------


用正则表达式来删除 必须要用 / / d 格式匹配

[root@xu opt]# sed -n "/xly/,/abc/d" /etc/passwd




从第 2行开始(包括第2行)在往下删去2行

[root@xu ~]# sed "2,+2d" /etc/passwd


从2开始 删掉第一个是5的倍数行之间的行

[root@xu ~]# sed "2,~5d" /etc/passwd



删除偶数行(能除2的) 每2行删去第1行

[root@xu ~]# sed "1~2d" /etc/passwd


--------------------------------


替换 s “ s / / /gi“ g全局 i忽略大小写


[root@xu ~]# sed -n "s/a/b/gi" /etc/passwd


替换多个范围的 1行到3行

[root@xu ~]# sed -n "1,3s/a/b/gi" /etc/passwd


------


用正则 修改 用// 圈起来


把所有的数字改成字母

[root@xu ~]# sed -n "s/[0-9]/b/g" /etc/passwd


把包含有字母都删除

[root@xu ~]# sed -n "s/[a-z]//g" /etc/passwd


修改运行级别到3


[root@xu ~]# grep initdefault /etc/inittab


# 0 - halt (Do NOT set initdefault to this)

# 6 - reboot (Do NOT set initdefault to this)

id:5:initdefault:

[root@xu ~]# sed "/id/s/[0-6]/3/" /etc/inittab

#



在sed 的正则表达式里

-r

在正则表达式里 不用转意符 可以用-r 解决

[root@xu ~]# sed -r "/id/s/[0-6]{2}/3/" /etc/inittab



把/etc/passwd 文件里 没一行的第一个字符删掉

[root@xu ~]# sed -n "s/.//" /etc/passwd


把/etc/passwd 文件里 每一行的第一个和最后一个字符删掉





() 保存所匹配的字符 & 代替你查找的所有字符


每一个模式 是\1 \2 ...\9 每个模式对应前面的第几个()里的内容




出现在第一个 是开头 第二个是去掉最后一个的所有内容 的三个是最后一个内容

[root@xu ~]# sed -n -r "s/(.)(.*)(.)/\2/" /etc/passwd



修改ip最后的地址为254

[root@xu opt]# sed -r "s/(192.168.1.)(.*)/\1254/" ip.txt

192.168.1.254

192.168.1.254

192.168.1.254

192.168.1.254

192.168.1.254

192.168.1.254

192.168.1.254



把一个文件所有的大写 字母加上个小括号

[root@xu opt]# sed "s/[A-Z]/(&)/g" /etc/passwd


-----------------------

添加 a 默认的是每一行都添加一个


[root@xu opt]# sed -r "aGATEWAY=192.168.1.254" ifcfg-eth0

指定位置添加

[root@xu opt]# sed -r "1aGATEWAY=192.168.1.254" ifcfg-eth0




插入 i 插入到上面

[root@xu opt]# sed -r "1iGATEWAY=192.168.1.254" ifcfg-eth0




替换 c

第一行替换成GATEWAY....

[root@xu opt]# sed -r "1cGATEWAY=192.168.1.254" ifcfg-eth0



导入 r

把有a里的内容 导入到搜索出来1这行的内容的下面

把只要有1的行导入到a.txt里去

什么内容到导入到那里原文件

[root@xu opt]# sed -r "/1/r a.txt" 1.txt





导出 w

把有a的内容写到 1.txt里去 会覆盖之前的内容

把只要有a的导出到1.txt里

[root@xu opt]# sed -r "/a /w 1.txt" a.txt



------------


命令组

root@xu opt]# sed -n "2=;2p" a.txt

[root@xu opt]# sed -n "2{=;p}" a.txt


-e 如果有多个命令要执行的时候 可以加多个 -e 连起来 作成一个命令组

[root@xu opt]# sed -n -e "2=" -e "2p" a.txt、


除了第一行剩下的内容 取反

[root@xu opt]# sed -n "1 ! p" 2.txt



-------------------------

删出 d

[root@xu opt]# sed "d" a.txt


4n 把当前行 读出下一行 也就是 第4行不读 其他的 都读

[root@xu opt]# sed "4n;d" a.txt


-----------------


sed 复制

有俩个空间 sed的缓存

一个是模式空间 里面不存数据

当sed 处理一个流的时候 他会把他放到 里买 处理的时候拿进来 处理完了拿出去

二个是保持空间

默认情况下什么都没有 只有一个换行符\n


俩个选项

剪贴


H 追加导入到保持空间

h 覆盖导入到保持空间



[root@xu opt]# sed "1h;1d;5G" 2.txt


粘贴


G 追加粘贴到模式空间

g 覆盖粘贴到模式空间


[root@xu opt]# sed "1H;5G" 2.txt

[root@xu opt]# sed "1,3h;5G" 2.txt


-------------------------------------

awk 单独的一门语言 也是逐行处理


awk 的格式


awk [选项] “动作” 文件名

命令 | awk [选项] “动作” 文件名


awk -F":" '{print FILENAME}'


位置变量 第一列的值就是$1

$1 。。。$n 分割


$0 代表整个行的信息


位置变量

[root@xu opt]# awk -F":" '{print $1,$2}' /etc/passwd

[root@xu opt]# head /etc/passwd | awk -F":" '{print $1,$2}'


打印当前文件名 FILENAME

[root@xu opt]# awk -F":" '{print FILENAME}' /etc/passwd

打印分割类的个数 NF

[root@xu opt]# awk -F":" '{print NF}' /etc/passwd


当前处理行的行数 NR

[root@xu opt]# awk -F":" '{print NR}' /etc/passwd

处理当前行在文件内的行数FNR

[root@xu opt]# awk -F":" '{print FNR}' /etc/passwd


AWK的默认分隔符 为空格 和 tab键默认时可以省略 -F


[root@xu opt]# head -3 /etc/passwd | awk -F":" '{print $1,$3,$4}'

root 0 0

bin 1 1

daemon 2 2



-------------

awk 处理数据的时候顺序是三个 行前 行中 行后

行前

BEGIN{}

[root@xu opt]# awk -F":" 'BEGIN{print NR}' /etc/passwd

0



行中

{}

[root@xu opt]# awk -F":" '{print NR}' /etc/passwd




行后的

END{}

[root@xu opt]# awk -F":" 'END{print NR}' /etc/passwd

73


-------------------



用正则


awk '条件{print NR}'


~ 匹配

!~ 不匹配


匹配是root的

[root@xu opt]# awk -F ":" '$1~/root/{print $1}' /etc/passwd

root


比较符号


= == != > >= < <=


逻辑比较 || &&



运算符号

+ - * / % += -= *= /=


i=5


i+=2

i=i+2


i=7


---------------------

awk流程控制

流程控制

分支结构

if (条件) 动作

若有多个动作,则要用大括号将动作体包含起来

if (条件) {动作 1;动作 2}

# awk -F : '{if ($1 == "root") print $1}' /etc/passwd

# awk -F: '{if ($1 == "root") {print $1;print $6} }' /etc/passwd

if (条件 1)

动作 1

else

动作 2

# awk -F: '{if ($1 == "root"){print $1}else print $6}' /etc/passwd

# awk -F: '{if ($1 == "root") print $1;else print $6}' /etc/passwd

上面两个命令是等价的 ,要么用分号隔开 ,表示第一个动作体的结束 ,要么将动作体用大括号定位范围

if (条件 1)

动作 1

else if(条件 2)

动作 2

else if(条件 3)

动作 3

else

动作 4

# awk -F: '{if ($1 == "root") print $1;else if ($1 == "seker") print $6;else if ($1 == "zorro") print $7;else print

NR}' /etc/passwd

条件 ? 动作 1 : 动作 2

Expr ? action1 : action2

# awk -F: '{print ($3<5?$1:$3)}' /etc/passwd

读前处理和读后处理

# awk -F: 'BEGIN {print NR,NF}' /etc/passwd

00

# awk -F: 'END {print NR,NF}' /etc/passwd

46 7

# awk -F: 'BEGIN{i=1} {i++} END {print i}' /etc/passwd

47

练习 找出普通用户的用户名并统计数量

#awk -F: 'BEGIN{i=0} $3>=500 {print $1;i++} END {print i}' /etc/passwd

循环语句

while(条件) {

动作

条件运算

}

# awk -F: '{while($3<3) {print $3,$1;$3++}}' /etc/passwd

BEGIN

BEGIN 块可以独立使用,不需要引入文件

# awk 'BEGIN{i=1;while(i<100) {print i;i++}}'

练习 打印 100 以内的偶数

# awk 'BEGIN{i=1;while(i<100) {if (i%2==0) print i;i++}}'

x=1

do {

动作 1

x++

} while (x<5)

# awk 'BEGIN{i=5;do{print i;i++}while(i<10)}'

# awk 'BEGIN{i=5;do{print i;i++}while(i<1)}'

for(预置;条件;递增) {

动作

}

# awk 'BEGIN {for (x=1;x<=4;x++) print x }'

输出样式

# awk -F: '{printf "%-10s %-10d %s\n",$1,$3,$7}' /etc/passwd

%s 字符类型, %d 数值类型

printf 默认是不输出换行的所以要加\n

10 和 7 是偏移量

默认是右对齐,所有加个- 就是左对齐,就是把不足的位数用空格填充

注意:格式与输出列之间要有逗号

跳转语句

break 跳出循环

# awk 'BEGIN {for(x=1;x<5;x++) {if (x==3) break;print x }}'

1

2

continue 在达到循环底部之前终止当前循环 从新开始下一次循环

# awk 'BEGIN {for(x=1;x<5;x++) {if (x==3) continue;print x }}'

1

2

4

next 读入下一行 同时返回脚本顶部 这样可以避免对当前行执行其他操作

# awk -F: 'NR > 5 {next} {print $1} END {print NR}' /etc/passwd

root

bin

daemon

adm

lp

46

46

#

exit 使读取动作终止 并将控制移动到 END,如果没有 END 则终止脚本

# awk -F: 'NR > 5 {exit} {print $1} END {print NR}' /etc/passwd

root

bin

daemon

adm

lp

6

[root@svr10 pub]# cat -b u.txt

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

2 bin:x:1:1:bin:/bin:/sbin/nologin

3 daemon:x:2:2:daemon:/sbin:/sbin/nologin

4 adm:x:3:4:adm:/var/adm:/sbin/nologin

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

9 mail:x:8:12:mail:/var/spool/mail:/sbin/nologin

10 news:x:9:13:news:/etc/news:

[root@svr10 pub]# awk -F: 'BEGIN{i=1}NR>5 {next}{print i++}END{print NR}' u.txt

1

2

3

4

5

10

[root@svr10 pub]# awk -F: 'BEGIN{i=1}NR>5 {exit}{print i++}END{print NR}' u.txt

1

2

3

4

5

6

[root@svr10 pub]#

数组

自定义数组

# awk 'BEGIN {ary[1]="seker";ary[2]="zorro";print ary[1],ary[2]}'

seker zorro

# awk 'BEGIN {ary[1]="seker";ary[2]="zorro";for(i in ary) print ary[i]}'

seker

zorro

#

循环产生数组和取出数组

# awk 'BEGIN{n=5;for (i=1;i<=n;i++) ary[i]=i+100;for(m in ary) print m,ary[m]}'

4 104

5 105

1 101

2 102

3 103

#

# awk -F: '{ary[NR]=$1} END {for(i in ary) print i,ary[i]}' /etc/passwd

1 root

2 bin

3 daemon

4 adm

5 lp

6 sync

7 shutdown

8 halt

9 mail

# awk -F: '{ary[$3]=$1} END {for(i in ary) print i,ary[i]}' /etc/passwd

10 uucp

11 operator

12

12 games

13 gopher

14 ftp

32 rpc

37 rpm