shell 脚本扩展

发布时间:2020-07-16 22:54:05 作者:lyanhong
来源:网络 阅读:208

一、            正则表达式:

 

正则表达式(或称Regular Expression,简称RE)就是由普通字符(例如字符 a z)以及特殊字符(称为元字符)组成的文字模式。

 

该模式描述在查找文字主体时待匹配的一个或多个字符串。

 

正则表达式作为一个模板,将某个字符模式与所搜索的字符串进行匹配。简单的说,正则表示式就是处理字符串的方法,它是以行为单位来进行字符串的处理行为,正则表示式通过一些特殊符号的辅助,可以让使用者轻易的达到搜寻/删除/取代某特定字符串的处理程序。vimgrepfindawksed等命令都支持正则表达式。

常用正则表达式:

1.代表任意单个字符,如:/l..e/与包含一个l,后跟两个字符,然后跟一个e的行相匹配

 

2^代表行的开始。 ^love 如:与所有love开头的行匹配

 

3$代表行的结束。love$ 如:与所有love结尾的行匹配

那么‘^$’ 就表示空行

 

4[]匹配括号中的字符之一

[abc]     匹配单个字符abc

[123]     匹配单个字符123

[a-z]     匹配小写字母a-z之一

[a-zA-Z]    匹配任意英文字母之一

[0-9a-zA-Z]匹配任意英文字母或数字之一

注意:上面标红色的单个和之一,不管[]里面多复杂,它的结果都是一个字符!

 

可以用^标记做[]内的前缀,表示[]内的字符之外的其他字符(即匹配不在此括号中的任何字符)。比如搜索oo前没有g的字符串的行. 应用 '[^g]oo' 作搜索字符串,^符号如果出现[]的起始位置表示否定,但是[]的其他位置是普通字符[^ab^c] 除了ab^c的其他任意单个字符

 

5、* 用于修饰前导字符,表示前导字符出现0或任意多

如:'a*grep'匹配所有0个或多个a后紧跟grep的行。.*”表示任意字符串

 

6\?用于修饰前导字符,表示前导字符出现0或1

a\? 匹配0或1个a

 

7\+用于修饰前导字符,表示前导字符出现1或多

a\+ 匹配1或多个a

 

8\{n,m\} 用于修饰前导字符,表示前导字符出现n至m次 (n和m都是整数,且n<m)

a\{3,5\} 匹配3至5个连续的a

 

\{n,m\}还有其他几种形式:

\{n\} 连续的n前导字符

\{n,\} 连续的至少n个前导字符

9、\ 用于转义紧跟其后的单个特殊字符,使该特殊字符成为普通字符

 

如:^\.[0-9][0-9]以一个句点和两个数字开始

 

 

例如:

a* 匹配连续的任意(也包括0)个a

a\? 匹配0或1个a

a\+ 匹配1或多个a

a\{3,5\} 匹配3至5个连续的a

\.* 匹配0或多个连续的.  \.表示普通字符句点

 

10|表示  如:a|b|c 匹配abc。如:grep|sed匹配grepsed

 

11(),将部分内容合成一个单位组,比如要搜索 glad good可以如下 'g(la|oo)d'

二、grep命令的用法

grep global search regular expression(RE) and print out the line,全面搜索正则表达式并把行打印出来)是一种强大的文本搜索工具,它能使用正则表达式搜索文本,并把匹配的行打印出来.

 

参数:

1. -ANUM--after-context=NUM   除了列出符合行之外,并且列出后NUM行。

:   $ grep –A1 panda file  (file中搜寻有panda样式的行,并显示该行的后1)

 

2. -BNUM--before-context=NUM  -A NUM 相对,但这此参数是显示除符合行之外并显示在它之前的NUM行。如:    (file中搜寻有panda样式的行,并显示该行的前1)

$ grep -B 1 panda file

 

3 -C[NUM], -NUM, --context[=NUM]  列出符合行之外并列出上下各NUM行,默认值是2

 

:   (列出file中除包含panda样式的行外并列出其上下2)(若要改变默认值,直接改变NUM即可)

$ grep -C[NUM] panda file

 

4 -c, --count  不显示符合样式行,只显示符合的总行数。若再加上-v,--invert-match,参数显示不符合的总行数

5-i--ignore-case 忽略大小写差别

 

6-n--line-number 在匹配的行前面打印行号

 

7-v--revert-match  反检索,只显示不匹配的行

 

8、精确匹配:

例如在抽取字符串“ 48”,返回结果包含诸如484483等包含“48”的其他字符串,实际上应精确抽取只包含48的各行。

 

使用grep抽取精确匹配的一种有效方式是在抽取字符串后加\>。假定现在精确抽取48

方法如下:

#grep '48\>' filename

 

9-s 不显示不存在或无匹配文本的错误信息

如:执行命令grep "root" /etc/password,因为password文件不存在,所以在屏幕上输出错误信息,若使用grep命令-s开关,可屏蔽错误信息

 

要用好grep这个工具,其实就是要写好正则表达式,所以这里不对grep的所有功能进行实例讲解,只列几个例子,讲解一个正则表达式的写法。

 

$ ls -l | grep  '^d'
通过管道过滤ls -l输出的内容,只显示以d开头的行。

 

$ grep  'test'  d*
显示所有以d开头的文件中包含test的行。

 

$ grep  'test'  aa  bb  cc
显示在aabbcc文件中匹配test的行。

 

$ grep  '[a-z]\{5,\}'  aa
显示所有包含每个字符串至少有5个连续小写字符的字符串的行。

 

$grep  ‘t[a|e]st’ filename

显示包含testtast的所有行。

 

$grep  '\.$' filename

显示以.为结尾的所有行。

 

三、sed命令的用法

sed是一种在线编辑器,它一次处理一行内容。处理时,把当前处理的行存储在临时缓冲区中,称为模式空间pattern space),接着用sed命令处理缓冲区中的内容,处理完成后,把缓冲区的内容送往屏幕。接着处理下一行,这样不断重复,直到文件末尾。文件内容并没有改变,除非你使用重定向存储输出。

 

sed的基本命令

1替换: s命令
        1.1 基本用法

  如: sed 's/day/night/'<old >new
 
该例子将文件 old 中的每一行第一次出现的 day 替换成 night, 将结果输出到文件 new

 

 s            " 替换 " 命令
  /../../     
分割符 (Delimiter)
  day         
搜索字符串
  night       
替换字符串
 
其实 , 分割符 "/" 可以用别的符号代替 , 比如 ",", "|" .
 
如:sed's/\/usr\/local\/bin/\/common\/bin/'<old >new
 
等价于 sed 's_/usr/local/bin_/common/bin_' <old >new
 
显然 , 此时用 "_" 作分割符比 "/" 好得多

 

1.2 & 表示匹配的字符串

有时可能会想在匹配到的字符串周围或附近加上一些字符 .
 
如: sed 's/abc/(abc)/' <old >new

 

 该例子在找到的 abc 前后加上括号 .
 
该例子还可以写成 sed's/abc/(&)/' <old >new

 

 下面是更复杂的例子 :
 sed 's/[a-z]*/(&)/' <old >new

 

sed 默认只替换搜索字符串的第一次出现 , 利用 /g 可以替换搜索字符串所有

$ sed's/test/mytest/g' example-----在整行范围内把test替换为mytest。如果没有g标记,则只有每行第一个匹配的test被替换成mytest

$ sed's/^192.168.0.1/&localhost/' example-----&符号表示替换字符串中被找到的部份。所有以192.168.0.1开头的行都会被替换成它自已加 localhost,变成192.168.0.1localhost

$ sed 's#10#100#g' example-----不论什么字符,紧跟着s命令的都被认为是新的分隔符,所以,“#”在这里是分隔符,代替了默认的“/”分隔符。表示把所有10替换成100

如果需要对同一文件或行作多次修改,可以使用 "-e" 选项

 

 

2删除行:d命令

 

3.增加行:a命令(在指定的行后新增)或i命令(在指定的行前新增)

a的后面可以接字符串,而这些字符串会在新的一行出现

4、取代行:c命令

c的后面可以接字符串,这些字符串可以取代n1,n2之间的行

5、打印:p命令

sed'/north/p' datafile 默认输出所有行,找到north的行重复打印

 

sed n '/north/p'datafile 禁止默认输出,只打印找到north的行

 

nl/etc/passwd | sed -n '5,7p' 仅列出/etc/passwd文件中的第57行内容

 

6.扩展:

调用sed有三种方式:

l  在命令行键入命令

l  sed命令插入脚本文件,然后调用sed

l  sed命令插入脚本文件,并使sed脚本可执行。

A、使用sed命令行格式为:

sed [选项] sed命令 输入文件。

记住在命令行使用sed命令时,实际命令要加单引号。sed也允许加双引号。

 

B、使用sed脚本文件,格式为:

sed [选项] -f sed脚本文件  输入文件

 

C、要使用第一行具有sed命令解释器的sed脚本文件,其格式为:

sed脚本文件 [选项]  输入文件

 

不管是使用shell命令行方式或脚本文件方式,如果没有指定输入文件, sed从标准输入中接受输入,一般是键盘或重定向结果。

 

sed选项如下:

-f, --filer=script-file 引导sed脚本文件名

四、printf命令:

printf是一个把从标准输入的字符按照你所要求的格式输出到标准输出即屏幕的命令.在很多时候,我们可能需要将自己的数据给他格式化输出的。

五、awk命令:

awk也是一个数据处理工具!相较于 sed 常常作用于一整个行的处理, awk 则比较倾向于一行当中分成数个字段来处理。

 

.awk语言的最基本功能是在文件或字符串中基于指定规则来分解抽取信息,也可以基于指定的规则来输出数据。其实他更像一门编程语言,他可以自定义变量,有条件语句,有循环,有数组,有正则,有函数等。

 

有三种方式调用awk

 

1.命令行方式

awk  [-F field-separator]  'commands' input-files

其中,[-F域分隔符]是可选的,因为awk使用空格或tab键作为缺省的域分隔符,因此如果要浏览域间有空格的文本,不必指定这个选项,如果要浏览诸如passwd文件,此文件各域以冒号作为分隔符,则必须指明-F选项,如:awk -F: 'commands' input-file

:linux系统中用环境变量IFS存储分隔符,但根据实际应用也可以改变IFS的值.

2shell脚本方式

将所有的awk命令插入一个文件,并使awk程序可执行,然后awk命令解释器作为脚本的首行,以便通过键入脚本名称来调用。

相当于shell脚本首行的:#!/bin/sh可以换成:#!/bin/awk

 

3.将所有的awk命令插入一个单独文件,然后调用:

Awk    -f   awk-script-file         input-files

其中,-f选项加载awk-script-file中的awk脚本,input-files跟上面的是一样的。

 

awk的模式和动作

 

任何awk语句都由模式和动作组成(awk_pattern { actions })。
在一个awk脚本中可能有许多语句。

模式部分决定动作语句何时触发及触发事件。处理即对数据进行的操作。如果省略模式部分,动作将时刻保持执行状态。即省略时不对输入记录进行匹配比较就执行相应的actions。

 

模式可以是任何条件语句或正则表达式等。awk_pattern可以是以下几种类型:

 

1) 正则表达式用作awk_pattern:/regexp/

例如:awk '/^[a-z]/' input_file

2) 布尔表达式用作awk_pattern,表达式成立时,触发相应的actions执行。

表达式中可以使用变量(如字段变量$1,$2等)和/regexp/

② 布尔表达式中的操作符:

 

关系操作符: <> <= >= == !=
匹配操作符: value ~ /regexp/ 如果value匹配/regexp/,则返回真

value!~ /regexp/ 如果value不匹配/regexp/,则返回真
例如: awk '$2 > 10 {print "ok"}' input_file
      awk '$3 ~ /^d/ {print"ok"}' input_file

③ &&(与) 和 ||(或) 可以连接两个/regexp/或者布尔表达式,构成混合表达式。!(非) 可以用于布尔表达式或者/regexp/之前。

 

例如: awk '($1< 10 ) && ($2 > 10) {print "ok"}' input_file
      awk '/^d/ || /x$/ {print"ok"}' input_file

模式包括两个特殊字段 BEGINEND。使用BEGIN语句设置计数和打印头。BEGIN语句使用在任何文本浏览动作之前,之后文本浏览动作依据输入文本开始执行。END语句用来在awk完成文本浏览动作后打印输出文本总数和结尾状态标志。

 

实际动作在大括号{ }内指明。动作大多数用来打印,但是还有些更长的代码诸如i f和循环语句及循环退出结构。如果不指明采取动作,awk将打印出所有浏览出来的记录。

 

awk执行时,其浏览域标记为$1$2...$n。这种方法称为域标识。使用这些域标识将更容易对域进行进一步处理。

 

使用$1 , $3表示参照第1和第3域,注意这里用逗号做域分隔。如果希望打印一个有5个域

的记录的所有域,不必指明$1 , $2 , $3 , $4 , $5,可使用$0,意即所有域。

为打印一个域或所有域,使用print命令。这是一个awk动作

 

awk的运行过程:

 如果BEGIN 区块存在,awk执行它指定的actions。

  awk从输入文件中读取一行,称为一条输入记录。(如果输入文件省略,将从标准输入读取)

  awk将读入的记录分割成字段,将第1个字段放入变量$1中,第2个字段放入$2,以此类推。$0表示整条记录。

  把当前输入记录依次与每一个awk_cmd中awk_pattern比较,看是否匹配,如果相匹配,就执行对应的actions。如果不匹配,就跳过对应的actions,直到比较完所有的awk_cmd。

  当一条输入记录比较了所有的awk_cmd后,awk读取输入的下一行,继续重复步骤,这个过程一直持续,直到awk读取到文件尾。

  当awk读完所有的输入行后,如果存在END,就执行相应的actions。

六、函数及脚本的综合应用

 

1shell函数:shell允许将一组命令集或语句形成一个可用块,这些块称为shell函数。

 

函数由两部分组成:函数标题、函数体。

 

标题是函数名。函数体是函数内的命令集合。

 

标题名应该唯一;如果不是,将会混淆结果,因为脚本在查看调用脚本前将首先搜索函数调用相应的shell

 

2定义函数的格式为:

函数名(){

命令1

...

}

如果愿意,可在函数名前加上关键字function,这取决于使用者。

function函数名()

{

  命令1

  ...

}

 

注:

1、basename命令能够从路径中分离出文件名。通常用于shell脚本中

2、shift语句用于迁移位置变量,将 $1~$9 依次向左传递

例如,若当前脚本程序获得的位置变量如下:

$1=file1、$2=file2、$3=file3、$4=file4

则执行一次shift命令后,各位置变量为:

$1=file2、$2=file3、$3=file4

再次执行shift命令后,各位置变量为:

$1=file3、$2=file4

推荐阅读:
  1. shell脚本
  2. shell 案例

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

shell he hell

上一篇:原生nodejs使用websocket

下一篇:Hadoop之生态系统

相关阅读

您好,登录后才能下订单哦!

密码登录
登录注册
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》