Linux 脚本设计3——三种流程控制语句(顺序语句、分支语句、循环语句)

时间:2022-09-08 16:38:23

1.Linux流程控制语句                                              

  控制语句就是控制语句往哪走怎么走,linux控制语句与其他语言一样有三种:条件语句,分支语句和循环语句,这三种语句也反应了现实中的逻辑。

1.1条件语句                           

if,if else,if elif语句

上述都用fi结束

条件语句:先进行逻辑条件判断,再决定程序执行哪一种情况

表示逻辑条件有如下方式:

1、  单一变量

2、  表达式

注意:每一个if(包括elif)后面都要有一个then,并且最后以fi结束

1.1.1 if语句使用语法

if [condition]

  then

      instruction

fi

 

1.1.2 if…else 语句使用语法

if [condition]

 then

   instruction

else

   instruction

fi

1.1.3 if …elif语句使用语法

if [condition1]

  then

      instruction1

elif[condition2]

  then

      instruction2

fi

上述三种if语句的使用区别:

处理能力

A:if…then…fi处理能力最末

B:if…then…else…fi处理能力次之

C:if…then …elif...then…elif…then…elif…then fi的处理能力最强

说明:

A只能处理一种情况,如果条件满足,就处理,如果不满足,不做任何处理,就是不输出别的信息到屏幕或文件,因为C只有一种分支。

B只能处理两种情况,一种是if后面的,一种是除了if以外的所有情况,也就是if条件的补集对应的情况都用else表示了

C的处理和java中的switch语句是一样的,假设一个问题有10种情况,那么第一个if代表一种情况,其他的9个elif,每一个elif代表一种情况,共计9种

#!/bin/bash

echo "Input your mark"

read mark

if [ $mark -gt 90 ]

   then

        echo "Talent"

fi

 

 

#!/bin/bash

echo "Input your mark"

read mark

if [ $mark -gt 90 ]

   then

        echo "Talent"

else

        echo "failed"

fi

 

#!/bin/bash

echo "Input your mark [0,100]:"

read mark

if [ $mark -ge 0 ] && [ $mark -lt 60 ]

   then

        echo "You'd better prepare to review your lesson for makeup"

elif [ $mark -eq 60 ]

   then

        echo "one more is waste,one less is hard,60 is your god"

elif [ $mark -gt 60 ] && [ $mark -lt 70 ]

   then

        echo "Qualified and Eligible"

elif [ $mark -ge 70 ] && [ $mark -lt 80 ]

   then

        echo "Nice"

elif [ $mark -ge 80 ] && [ $mark -lt 90 ]

   then

        echo "Excelent"

elif [ $mark -ge 90 ] && [ $mark -le 100 ]

   then

        echo "Talent"

elif [ $mark -lt 0 ]

   then

        echo "Input the right format mark"

fi

 

 

1.2 case语句                        

本章包括

1、使用case语句

2、使程序至少程序一条case语句

3、case语句在用户界面中的作用

前面已经知道if条件句的三种情况,已经说明了方法C的处理和case是等价的。由于C可读性较差,用case则显得清晰的多。介绍case语句。

case的使用语法

case $变量名 in

变量值1)

         处理语句(不包含分号)

;;

变量值2)

         处理语句(不包含分号)

;;

….

*)

         处理语句(不含分号)

//没有分号

esac

 

说明:

1、变量值就是变量名能取到的所有条件,变量值可以是一个单值,也可以是一个区间

每一个逻辑分支的处理语句不包含分号,但是逻辑分支下面包含两个分号

2、最后以esac结束,作为结束关键词。

3、case后面的变量名带有in关键字,表示变量名在如下的变量值中

4、变量值后面跟一个小括号).

5、无论什么时候引用某个变量,必须加上美元符号$

6、如果不能匹配所列的值,使用通配符*放在变量值的最后位置,加上一个),然后方式处理语句,注意通配符*)的处理语句下面是没有两个分号的。切记*是通配所有情况的,如果把*)放在第一个逻辑分支,那么它将会覆盖后面的所有分支,也就是说此时的case语句相当于只有一个逻辑分支,这个逻辑分支是*)的,它已经覆盖了后面的所有逻辑分支。

case语句在用户界面的作用

由于linux shell与用户的交互并不是像windows那样有个图形界面的对话框,而是使用一种字符菜单形式与用户交互,因此,每一种情况可以用case来处理。

将菜单与case的分支进行对应就可以建立linux程序的菜单处理。

语句的嵌套

  在上述提到的if及case语句中,每一个instruction都可以用一个更小的if及case程序模块来代替,这种情况就是语句的嵌套。

 

1.3 linux循环控制语句          

while循环使用语法

while(condition)

         do

                   instruction

done

 

注意:这个循环的的终结词是用done,而且多了do,这些与C的循环是有区别的。

可以将循环用于菜单、计时循环的设计

死循环:while循环条件永远为真,程序永不停止(程序执行外在环境没有受影响的条件下)的执行循环体,永远不会停止执行,这种循环称为死循环。

for in循环

使用语法:

for 变量名 in 取值列表

do  

         instruction

done

 

注意这里的变量名不需要加上双引号,取值列表用空格隔开

这个循环的含义是:变量名取取值列表中的每一个值后,然后执行循环体,取值列表有100个值,则执行循环列表一百次。

例如

#!/bin/bash

for name in hola hello bonjour

    do

        echo "Buenas noches,$name !"

done

 

Buenas noches,hola !

Buenas noches,hello !

Buenas noches,bonjour !

这里面name取了取值列表的每一个值,然后执行循环体

变量值表也称为单词表,如果变量值中含有空格符合,需用双引号括起来,变量值之间仍然用空格隔开,变量值中没有双引号的不需要用双引号,有双引号的单词和没有双引号的单词可以放在一起。

#!/bin/bash

#declare name="hola"

for name in "hola holala" hello bonjour

    do

        echo "Buenas noches,$name !"

done

 

1.4 循环嵌套和快速退出         

嵌套循环,一个循环内部包含别的循环,换种说法也就是一个循环外还有循环,循环是多层的。

写循环嵌套时候,必须注意的是while和for in两种循环do 和done都是配套的。

中止循环

break循环用于跳出一个循环设计的所有次循环,continue用于跳出一次循环,通常continue放的位置并非循环体的最后位置,这样跳出这次循环后就不用执行continue语句后的其他语句。

注意使用赋值语句:

let a = “$a + 1”,这里面左边的变量没有美元符号,右边有,右边的双引号可以有,也可以没有

while循环条件中括号与变量要隔开,引用变量时候可以用双引号也可以不用,但是需要用美元符号,for in循环中,变量不需要美元符号。

 

2.函数                                                                       

一段功能单一的函数可以成为子程序,一个代码文件里面通常可以包含几个函数,但是只有一个主函数,也就是一个主函数和其他几个附属函数构成一个相对完善的功能。

函数:

函数的定义语法:

使用关键字

function  函数名

{

语句

}

 

注意:1、shell里面的大括号放的位置不像java里面放的那样,第一个大括号紧靠在函数名外面,必须另起一行开始放。

2、只有一个函数而不调用函数,即:不在某一行中写改函数名,是不执行定义的函数里面的语句的。如

#!/bin/bash

clear

function hola

{

echo "hello world"

}

 

hola

如果没有加粗的hola函数调用,虽然定义了hola函数,但是是不执行的。调用才执行。

2.1 函数的传参                     

位置参数:在shell语言中,使用$1表示第一个参数,$2表示第二个参数,以此类推,$0表示程序名称。$表示变量的引用,不能用其他字符如val等类似变量名称的字符来代替这些数字,这与java是有不同的。

#!/bin/bash

clear

function verify

{

if [ $1 -eq 747787 ]&&[ $2 -eq 1234 ]

   then

        echo "Verified"

   else

        echo "Not Verified"

fi

}

echo "Input your name"

read name

echo "Input your password"

read password

verify $name $password

 

符号$#表示的意思是传递的参数个数,可以用参数个数校验的方法来确定是否去调用某个函数

 

函数的共用

函数的共用就是把一个函数变成一个公共的函数,这样大家都能调用,使用关键字export

语法:export 函数名

 

函数的返回值:使用return关键词 return 可以设定返回一个数表示一段程序执行的结果,这个数是0——256,不能返回字符串。通常用返回0表示返回正常,返回非0表示返回错误的结果。返回的结果存放在变量$?中,可以用于case语句的处理。

 

3.数据文件程序的设计与打印                                         

两个符号:

1.>   重定向符号,这个符号前面已经说了,就是讲信息写入重定向符号右边的文件中,覆盖原文件信息内容。

2.>> 也是重定向符号,它是添加到文件末尾的符号,并不覆盖。

利用gawk查找文件中的信息

搜索语法

gawk ‘/ABC/{print $1,$2….$N}‘ 文件名

注意:

1.ABC是表示要搜索的内容,用两个正斜杠/包着

2.文件名表示要在其中搜索的文件,需要与最后一个单引号隔开

3.总的来说这个搜索表达式包括三个部分 gawk  正则表达式 文件名,正则表达式用单引号包着,与其他两个部分隔开。

这个搜索表达式默认是首列进行匹配的,也可以进行指定列的匹配

 

gawk  ‘$N~/^123*/{print $1,$2….$n}’ 文件名

$N表示对指定的列进行匹配,~表示匹配一个搜索表达式,如果是!~表示不匹配一个搜索表达式,找出不等于搜索表达式的记录。^表示进行一个列的开头进行匹配,*是通配符。

上述搜索命令右方加上>可以讲搜索到的内容重定向到一个文件中。这样就可以搜索到想要的,把不想要的过滤掉,也就相当于删除了不想要的内容。

 

打印内容:

lp命令,利用lp命令可以调用打印程序,将内容送到已经安装好的打印机中,进行打印

 

cat abc | lp,其中|是管道符号

 

管道符号与重定向符号的区别:

管道符号|与重定向符号>或>>区别是管道符号两边都是命令,它的本质是将一个命令的输出当作另一个命令的输入,而重定向符号左边是命令的输出,右边是文件,也就是输出的最终地址。

现在有这样的问题:搜索满足某个条件的信息,然后打印。可以这样做使用gawk来搜索,然后利用管道命令,将输出结果输送到打印机,即lp命令,也就是:

gawk 正则表达式 文件名 | lp

这样写有一个问题,就是每搜索一个结果就送到打印机,这样就需要反复执行gawk命令,这样做效率很低。可以先把搜索结果放在一个文件中,然后将整个文件送去打印。

gawk 正则表达式 文件名>新文件名

cat 新文件名|lp

这样会提高效率。

 

 

注释:

注释用于解释一个程序的目的或者某一个函数的用途,对于函数的注释无须多说,对于一个程序文件的注释,通常放在程序的开头,并且一个程序的注释至少包含如下7个部分。

#程序名称(就是程序文件的名称)

#程序的功能

#程序员的名称

#使用程序的潜在用户

#程序的设计日期

#程序最后一次修改日期

#最后一位修改的程序员名字

例如:

#Title:ForInDemo

#Purpose:a demo to show how to use for in sentence

#Programer:Jack Ren

#Primary User:programer

#Originally Written:13/04/13

#Last Changed:13/04/13

#Changed by:Jack Ren

注释使用#符号,也可以把这个#放在一个语句的开头,使语句失效

 

4.email程序设计                                                    

使用linux email邮件

每一台linux机器上都有一个linux账号,账号与主机名组合就是该email的账号。

比如说linux机器名是Landau,有一个linux账号是qq,那么该主机上的邮件账号是qq@Landau。

Linux email的邮件程序是mail

使用语法:mail 邮件地址 < 发送的内容文件

可以携带参数

-s  表示邮件的主题     mail –s “关于神经网络系统的设计” qq@Landau < neutralnetsystemdesign.txt

 

-c   表示明抄送,就是大家都可以看到的抄送地址 如果不带这个参数,发送给多个人时候,需要将邮件地址以空格隔开。 mail –c qq@Landau bob@Landau < 发送的内容文件,不带-c,所有人都在目的地址中,带-c,-c后面的人在抄送列表中,邮件的收件人可以看到所有的地址。

-b  表示密送,密送地址可以看到所有明送的抄送地址,但是看不到其他密送的地址。

mail –b qq@Landau < 发送的内容文件

 

三个参数可以同时使用,参数的作用范围,如果参数不冲突,则一直到邮件地址结束,如果有冲突,则到下一个参数开始前为止

 

Mail –s “你好” –c qq@Landau –b bob@Landau < nihao.txt

 

一个统计写的程序有多少行的程序代码

cat * >abc.txt

:set nu

就可以看到代码多少行了

 

5.一些有用的linux shell程序                                     

spell 拼写检查程序,能够对作用的文本内容进行检查

比如spell abc.txt,如果abc.txt中有错误的单词,将检查出来

spell的作用原理是将文本的单词与linux字典中的文件进行比较,如果字典中没有这个单词,那么就会显示出来,这其实是个缺陷,因为像地方名称、名字这些东西很可能不在linux字典文件中。

排序:sort

使用语法:sort abc.txt

sort默认的排序是升序。并且小写在大写前面,几个参数

-r reduce 表示降序

-f 表示忽略大小写

-n number 按照首字母数字排序

-m month 按照月份排序

比较文件差异:diff

diff file1 file2

hadoop@sdp15:~> diff hello.txt hello.sh

1,5c1,6

< 2Hello

< 1hola

< 4hello

< hello123

< hello1234

---

> #!/bin/bash

> clear

> echo"Welcome to my first linux program"

> echo"Enter your first name"

> read response

> echo "Hello

其中<表示只在file1中有的,>表示只在file2中有的,没有显示的表示是共同有的。

 

比较程序cmp

这个程序会比较每一个字符,与diff都是比较,但是比较的比较细致。

 

对大型文件的修改:sed程序

 

一般的文件编辑器工作是这样的,它将整个文件读入到系统的缓冲区,然后进行操作,如果文件太大,那么整个缓冲区将没法装下这个文件,也就无法继续操作。sed的工作原理不同,每次只读入一行数据,然后保存文件到一个后缀名为.tmp(其实任何一种后缀名都是可以的,后缀名其实没啥用,有没有都一样的,只是一种类型标识而已)中,而不是原文件中。这样可以先用sed修改文件,然后保存到原文件.tmp中,最后用mv命令修改原文件.tmp为原文件名称即可(会自动替换原文件的)

语法:

sed 正则表达式 原始文件 > 一个tmp文件中

重命名tmp文件

 

拆分大型文件split

语法:split –n filename

-n代表每一个文件行数,最后一个文件可能没有这么多行,就相当于一个除法一样,最后一个文件如果不是-n行数,就以余数行数为内容。拆开后的小文件以xaa,xab,xac。。。类推命名。split一个文件后,原文件依然存在而不被删除。

查找文件中的字符

grep命令

使用语法:grep 查找的内容 文件名

如果不确定在哪个文件,可以使用*进行通配

 

ftp命令 远程登录

使用语法:ftp  IP地址

输入后提示用户输入账号和密码,正确后就可以登录

注意:ftp支持的linux命令并不是和linux本机完全一致的,比如说ll在suse linux中可以用,但是在ftp模式下,用ll是无效的,可以用ls。

用ftp获取文件

ftp> get 文件名

获取的文件在当前本机的目录下

传递文件到对方机器上

使用命令

ftp>put 文件名

这时候文件就被传送到对方机器当前目录下

断开链接:直接输入bye即ftp>bye 即可

 

剪切 文件中的列

使用命令cut

cut –f 1,2(1,2表示列) 作用的文件名 > 剪切的列放置到新的文件中,如果不用重定向符号>此时会将剪切的在屏幕中显示

注意:linux命令是通过一个tab键来确定两个字符串之间是否是列的关系,如果两个字符串之间是一个空格,系统会认为是一列的。例如

Jim Green是一个field,而Jim      Green是两个fields

注意:每次启动一个程序时候都会启动一个shell

修改文件中的字符

命令tr

tr 正则表达式 <原始文件名>目的文件名

注意用了重定向符号