快速学习Bash

做者:Vamei 出处:http://www.cnblogs.com/vamei 严禁转载。html


Shell是Linux下经典的文本互动方式,而Bash是如今最经常使用的一种Shell。我在这里总结了Bash的要点知识。程序员

 

Shell综述

Linux图形化桌面算不上精美。幸亏,Linux提供了更好的与树莓派互动的方式:Shell。打开终端(Terminal),桌面上就会出现一个黑色背景的窗口,里面就运行着一个Shell。若是你敲击键盘,会发现字符会显示在$提示符的后面,造成一串文本形式的命令。所谓的Shell,就是运行在终端中的文本互动程序。Shell分析你的文本输入,而后把文本转换成相应的计算机动做。express

 

在后面的内容中,我将用$来表示Linux系统Shell的命令提示符。好比说输入date命令:编程

$date

date用于日期时间的相关功能。敲击回车键Enter后,Shell会显示出系统当前的时间。bash

 

Shell看起来简陋,但实际上比图形化桌面强大得多。它是Unix体系下的文本交互界面。你只须要用键盘来输入文本,就能够和操做系统交互。但这仍是不够具体。说到底,Shell实际上是一个运行着的程序。这个程序接收到你按下回车键之间的输入,就会对输入的文本进行分析。好比下面这个命令:app

$free -h

包括空格在内总共7个字符。Shell程序会经过空格,区分出命令的不一样部分。第一个部分是命令名。剩下的部分是选项和参数。在这个例子中,Shell会进一步分析第二个部分,发现这一部分的开头是"-"字符,从而知道它是一个选项。less

 

有了命令名,Shell下一步就要执行该命令名对应的动做。这听起来就像是在戏剧舞台上,演员按照脚本演戏。Shell命令能够分为以下三类:运维

  • Shell内建函数(built-in function)
  • 可执行文件(executable file)
  • 别名(alias)

Shell的内建函数是Shell自带的功能,而可执行文件是保存在Shell以外的脚本,提供了额外的功能。Shell必须在系统中找到对应命令名的可执行文件,才能正确执行。咱们能够用绝对路径来告诉Shell可执行文件所在的位置。若是用户只是给出了命令名,而没有给出准确的位置,那么Shell必须自行搜索一些特殊的位置,也就是所谓的默认路径。Shell会执行第一个名字和命令名相同的可执行文件。这就至关于,Shell帮咱们自动补齐了可执行文件的位置信息。咱们能够经过which命令,来肯定命令名对应的是哪一个可执行文件:编程语言

$which date

 

别名是给某个命令一个简称,之后在Shell中就能够经过这个简称来调用对应的命令。在Shell中,咱们能够用alias来定义别名:编辑器

$alias freak="free -h"

Shell会记住咱们的别名定义。之后我在这个Shell中输入命令freak时,都将等价于输入free -h。

 

在Shell中,咱们能够经过type命令来了解命令的类型。若是一个命令是可执行文件,那么type将打印出文件的路径。

$type date $type pwd

总的来讲,Shell就是根据空格和其余特殊符号,来让电脑理解并执行用户要求的动做。到了后面,咱们还将看到Shell中其余的特殊符号。

 

Shell的选择

Shell是文本解释器程序的统称,因此包括了不止一种Shell。常见的Shell有sh、bash、ksh、rsh、csh等。在树莓派中,就安装了sh和bash两个Shell解释器。sh的全名是Bourne Shell。名字中的玻恩就是这个Shell的做者。而bash的全名是Bourne Again Shell。最开始在Unix系统中流行的是sh,而bash做为sh的改进版本,提供了更加丰富的功能。通常来讲,都推荐使用bash做为默认的Shell。树莓派,以及其余Linux系统中普遍安装sh,都是出于兼容历史程序的目的。

 

咱们能够经过下面的命令来查看当前的Shell类型:

$echo $SHELL

echo用于在终端打印出文本。而$是一个新的Shell特殊符号。它提示Shell,后面跟随的不是通常的文本,而是用于存储数据的变量。Shell会根据变量名找到真正的文本,替换到变量所在的位置。SHELL变量存储了当前使用的Shell的信息你能够在bash中用sh命令启动sh,并能够用exit命令从中退出。

 

命令的选项和参数

咱们已经看到,一行命令里还能够包含着选项和参数。总的来讲,选项用于控制命令的行为,而参数说明了命令的做用对象。好比说:

$uname -m

 

在上面的命令中,选项-m影响了命令uname的行为,致使uname输出了树莓派的CPU型号。若是不是该选项的影响,uname输出的将是"Linux"。咱们不妨把每一个命令看作多功能的瑞士军刀,而选项让命令在不一样的功能间切换。由一个"-"引领一个英文字母,这成为短选项。多个短选项的字母能够合在一块儿,跟在同一个"-"后面。好比,下面的两个命令就等价:

$uname -m -r $uname -mr

 

此外还有一种长选项,是用"--"引领一整个英文单词,好比:

$date --version

上面的命令将输出date程序的版本信息。

 

若是说选项控制了瑞士军刀的行为,那么参数就提供了瑞士军刀发挥用场的原材料。就拿echo这个命令来讲,它能把字符打印到终端。它选择打印的对象,正是它的参数:

$echo hello

 

有的时候,选项也会携带变量,以便来讲明选项行为的原材料。好比:

$sudo date --set="1999-01-01 08:00:00"

选项"--set"用于设置时间,用等号链接的,就是它的参数。date会把日期设置成这一变量所表明的日期。若是用短选项,那么就要用空格取代等号了:

$sudo date -s "1999-01-01 08:00:00"

值得注意的是,Shell对空格敏感。当一整个参数信息中包含了空格时,咱们须要用引号把参数包裹起来,以便Shell能识别出这是一个总体。

 

所谓的选项和参数提供给命令的附加信息。所以,命令最终会拿这些字符串作什么,是由命令本身决定的。所以,有时会发现一些特异的选项或参数用法。这个时候,你就要从文档中寻找答案。

 

变量

咱们能够在Bash中输入一行的命令。Bash会把输入的命令转化为特定的动做。从这一节起,咱们将看到Bash的可编程性。Bash提供了某些相似于C语言那样的编程语法,从而容许你用编程的方式,来组合使用Linux系统。咱们首先看Bash用变量存储数据的能力。正如咱们在C语言中看到的,变量是内存中的一起空间,能够用于存储数据。咱们能够经过变量名来引用变量中保持的数据。借助变量,程序员能够复用出现过的数据。Bash中也有变量,但Bash的变量只能存储文本。

 

1)变量赋值

Bash和C相似,一样用“=”来表示赋值。好比:

$var=World

就是把文本World存入名为var的变量,即赋值。根据Bash的语法,赋值符号“=”的先后不留空格。赋值号右边的文本内容会存入赋值号左边的变量。

 

若是文本中包含空格,那么你能够用单引号或双引号来包裹文本。好比:

$var='abc bcd'

或者:

$var="abc bcd"

 

在Bash中,咱们能够把一个命令输出的文本直接赋予给一个变量:

$now=`date`

借助``符号,date命令的输出存入了变量now。

 

咱们还能够把一个变量中的数据赋值给另外一个变量:

$another=$var

  

2)引用变量

咱们能够用$var的方式来引用变量。在Bash中,所谓的引用变量就是把变量翻译成变量中存储的文本。好比:

$var=World $echo $var

 就会打印出World,即变量中保存的文本。

 

在Bash中,你还能够在一段文本中嵌入变量。Bash也会把变量替换成变量中保存的文本。好比: 

$echo Hello$var

 文本将打印出HelloWorld。

 

为了不变量名和尾随的普通文本混淆,咱们也能够换用${}的方式来标识变量。好比说:

$echo $varIsGood

 因为Bash中并无varIsGood这个变量,因此Bash将打印空白行。但若是将命令改成: 

$echo ${var}IsGood

 Bash经过${}识别出变量var,并把它替换成数据。最终echo命令打印出WorldIsGood。

 

在Bash中,为了把一段包含空格的文本当作单一参数,就须要用到单引号或双引号。你能够在双引号中使用变量。好比:

$echo "Hello $var"

 将打印Hello World。与此相对,Bash会忽视单引号中的变量引用,因此单引号中的变量名只会被当作普通文本,好比:

$echo 'Hello $var'

 将打印Hello $var。

 

数学运算

在Bash中,数字和运算符都被当作普通文本。因此你没法像C语言同样便捷地进行数学运算。好比执行下面的命令:

$result=1+2 $echo $result

Bash并不会进行任何运算。它只会打印文本“1+2”。

 

在Bash中,你还能够经过$(())语法来进行数值运算。在双括号中你能够放入整数的加减乘除表达式。Bash会对其中的内容进行数值运算。好比

$echo $((2 + (5*2)))

 将打印运算结果12。此外,在$(())中,你也可使用变量。好比: 

$var=1 $echo $(($var + (5*2)))

 将打印运算结果11。

 

你能够用Bash实现多种整数运算:

 加法:$(( 1 + 6 ))。结果为7。
 减法:$(( 5 – 3 ))。结果为2。
 乘法:$(( 2*2 ))。结果为4。
 除法:$(( 9/3 ))。结果为3。
 求余:$(( 5%3 ))。结果为2。
 乘方:$(( 2**3 ))。结果为8。

 

如今,你就能够把数学运算结果存入变量:

$result=$(( 1 + 2 ))

 

 

返回代码

在Linux中,每一个可执行程序会有一个整数的返回代码。按照Linux惯例,当程序正常运行完毕并返回时,将返回整数0。所以,C程序中返回0的语句,都出如今C程序中main函数的最后一句。例以下面的foo.c程序:

int main(void) { int a; int b; int c; a = 6; b = 2; c = 6/2; return 0; }

 这段程序能够正常运行。所以,它将在最后一句执行return语句,程序的返回代码是0。在Shell中,咱们运行了程序后,能够经过$?变量来获知返回码。好比:

$gcc foo.c $./a.out $echo $?

 

若是一个程序运行异常,那么这个程序将返回非0的返回代码。好比删除一个不存在的文件: 

$rm none_exist.file $echo $?

 

 在Linux中,能够在一个行命令中执行多个程序。好比:

$touch demo.file; ls;

 

在执行多个程序时,咱们可让后一个程序的运行参考前一个程序的返回代码。好比说,只有前一个程序返回成功代码0,才让后一个程序运行: 

$rm demo.file && echo "rm succeed"

若是rm命令顺利运行,那么第二个echo命令将执行。

 

还有一种状况,是等到前一个程序失败了,才运行后一个程序,好比:

$rm demo.file || echo "rm fail"

若是rm命令失败,第二个echo命令才会执行。

 

Bash脚本

你还能够把多行的Bash命令写入一个文件,成为所谓的Bash脚本。当Bash脚本执行时,Shell将逐行执行脚本中的命令。编写Bash脚本,是咱们开始实现Bash代码复用的第一步。

 

1)脚本的例子

用文本编辑器编写一个Bash脚本hello_world.bash: 

#!/bin/bash echo Hello echo World

 

脚本的第一行说明了该脚本使用的Shell,即/bin/bash路径的Bash程序。脚本正文是两行echo命令。运行脚本的方式和运行可执行程序的方式相似,都是: 

$./hello_world.bash

须要注意的是,若是用户不具备执行Bash脚本文件的权限,那么他将没法执行Bash脚本。此时,用户必须更换文件权限,或者以其余身份登陆,才能执行脚本。当脚本运行时,两行命令将按照由上至下的顺序依次执行。Shell将打印两行文本:

Hello

World

 

Bash脚本是一种复用代码的方式。咱们能够用Bash脚本实现特定的功能。因为该功能记录在脚本中,所以我能够反复地运行同一个文件来实现相同的功能,而不是每次想用的时候都要从新敲一遍命令。咱们看一个简单的Bash脚本hw_info.bash,它将计算机的信息存入到名为log的文件中:

#!/bin/bash echo "Information of Vamei's computer:" > log lscpu >> log uname –a >> log free –h >> log

  

2)脚本参数

和可执行程序相似,Bash脚本运行时,也能够携带参数。这些参数能够在Bash脚本中以变量的形式使用。好比test_arg.bash:

#!/bin/bash echo $0
echo $1
echo $2

 

在Bash中,你能够用$0、$一、$2……的方式,来得到Bash脚本运行时的参数。咱们用下面的方式运行Bash脚本:

$./test_arg.bash hello world

$0是命令的第一部分,也就是./test_arg.bash。$1表明了参数hello,而$2表明了参数world。所以,上面程序将打印:

./test_arg.bash

hello

world

 

若是变动参数,同一段脚本将有不一样的行为。这大大提升了Bash脚本的灵活性。上面的hw_info.bash脚本中,咱们把输出文件名写死成log。咱们也能够修改脚本,用参数做为输出文件的文件名:

#!/bin/bash echo "Information of Vamei's computer:" > $1 lscpu >> $1
uname –a >> $1
free –h >> $1

 

借助参数,咱们就能够自由地设置输出文件的名字:

$./hw_info.bash output.file

 

3)脚本的返回代码

和可执行程序相似,脚本也能够有返回代码。仍是按照惯例,脚本正常退出时返回代码0。在脚本的末尾,咱们能够用exit命令来设置脚本的返回代码。咱们修改hello_world.bash:

#!/bin/bash echo Hello echo World exit 0

其实在脚本的末尾加一句exit 0并没必要要。一个脚本若是正常运行完最后一句,会自动的返回代码0。在脚本运行后,咱们能够经过$?变量查询脚本的返回代码: 

$./hello_world.bash $echo $?

 

若是在脚本中部出现exit命令,脚本会直接在这一行中止,并返回该exit命令给出的返回代码。好比下面的demo_exit.bash:

#!/bin/bash echo hello exit 1
echo world

你能够运行该脚本,检查其输出结果,并查看其返回代码。

 

函数

在Bash中,脚本和函数有不少类似的地方。脚本实现了一整个脚本文件的程序复用,而函数复用了脚本内部的部分程序。一个函数能够像脚本一个包含多个指令,用于说明该函数若是被调用会执行哪些活动。在定义函数时,咱们须要花括号来标识函数包括的部分: 

#!/bin/bash function my_info (){ lscpu >> log uname –a >> log free –h >> log } my_info

  

脚本一开始定义了函数my_info,my_info是函数名。关键字function和花括号都提示了该部分是函数定义。所以,function关键字并非必须的。上面的脚本等效于:

#!/bin/bash my_info (){ lscpu >> log uname –a >> log free –h >> log } my_info

花括号中的三行命令,就说明了函数执行时须要执行的命令。须要强调的是,函数定义只是食谱,并无转化成具体的动做。脚本的最后一行是在调用函数。只有经过函数调用,函数内包含的命令才能真正执行。调用函数时,只须要一个函数名就能够了。

 

像脚本同样,函数调用时还能够携带参数。在函数内部,咱们一样能够用$一、$2这种形式的变量来使用参数:

#!/bin/bash function my_info (){ lscpu >> $1
uname –a >> $1
free –h >> $1 } my_info output.file my_info another_output.file

在上面的脚本中,进行了两次函数调用。函数调用时,分别携带了参数output.file和another_output.file。

 

跨脚本调用

在Bash中使用source命令,能够实现函数的跨脚本调用。命令source的做用是在同一个进程中执行另外一个文件中的Bash脚本。好比说,有两个脚本,my_info.bash和app.bash。脚本my_info.sh中的内容是: 

#!/bin/bash function my_info (){ lscpu >> $1
uname –a >> $1
free –h >> $1 }

 

脚本app.bash中的内容是: 

#!/bin/bash source my_info.bash my_info output.file

运行app.bash时,执行到source命令那一行时,就会执行my_info.bash脚本。在app.bash的后续部分,就可使用my_info.bash中的my_info函数。

 

 

逻辑判断

咱们已经介绍了函数和脚本两种组合命令的方式。这两种方式均可以把多行命令合并起来,组成一个功能单元。函数和脚本都实现了必定程度的代码复用。从这一节起,咱们将看到选择和循环两种语法结构,这两种语法结构能够改变脚本的运行顺序,从而编写出更加灵活的程序。Bash除了能够进行数值运算,还能够进行逻辑判断。逻辑判断是决定某个说法的真假。咱们在生活中很天然地进行各类各样的逻辑判断。好比“3大于2”这个说法,咱们会说它是真的。逻辑判断就是对一个说法判断真假。在Bash中,咱们能够用test命令来进行逻辑判断:

$test 3 -gt 2; echo $?

命令test后面跟有一个判断表达式,其中的-gt表示大于,即greater than。因为“3大于2”这一表达式为真,因此命令的返回代码将是0。若是表达式为1,那么命令的返回代码是1:

$test 3 -lt 2; echo $?

表达式中的-lt表示小于,即less than。

 

数值大小和相等关系的判断,是最多见的逻辑判断。除了上面的大于和小于判断,咱们还能够进行如下的数值判断:

  • 等于: $test 3 -eq 3; echo $? 
  • 不等于: $test 3 -ne 1; echo $? 
  • 大于等于: $test 5 -ge 2; echo $? 
  • 小于等于: $test 3 -le 1; echo $? 

 

Bash中最多见的数据形式是文本,所以也提供了不少关于文本的判断:

  • 文本相同: $test abc = abx; echo $? 
  • 文本不一样: $test abc != abx; echo $? 
  • 按照词典顺序,一个文本在另外一个文本以前: $test apple > tea; echo $? 
  • 按照词典顺序,一个文本在另外一个文本以后: $test apple < tea; echo $? 

 

Bash还能够对文件的状态进行逻辑判断:

  • 检查一个文件是否存在: $test –e a.out; echo $? 
  • 检查一个文件是否存在,并且是普通文件: $test –f file.txt; echo $? 
  • 检查一个文件是否存在,并且是目录文件: $test –d myfiles; echo $? 
  • 检查一个文件是否存在,并且是软链接: $test –L a.out; echo $? 
  • 检查一个文件是否可读: $test –r file.txt; echo $? 
  • 检查一个文件是否可写: $test –w file.txt; echo $? 
  • 检查一个文件是否可执行: $test –x file.txt; echo $? 

 

在作逻辑判断时,能够把多个逻辑判断条件用“与、或、非”的关系组合起来,造成复合的逻辑判断。 

! expression expression1 –a expression2 expression1 –o expression2

 

 

选择结构

逻辑判断能够得到计算机和进程的状态。进一步,Bash能够根据逻辑判断,让程序有条件地运行,这也就是所谓的选择结构。选择结构是一种语法结构,可让程序根据条件决定执行哪一部分的指令。最先的程序都是按照指令顺序依次执行。选择结构打破了这一顺序,给程序带来更高的灵活性。最简单的,咱们能够根据条件来决定是否执行某一部分程序,好比下面的demo_if.bash脚本:

#!/bin/bash var = `whoami` if [ $var = "root" ] then
 echo "You are root"
 echo "You are my God."
fi

这个脚本中使用了最简单的if结构。关键字if后面跟着[],里面是一个逻辑表达式。这个逻辑表达式就是if结构的条件。若是条件成立,那么if将执行then到fi之间包含的语句,咱们称之为隶属于then的代码块。若是条件不成立,那么then的代码块不执行。这个例子的条件是判断用户是否为root。所以,若是是非root用户执行该脚本,那么Shell不会打印任何内容。

 

咱们还能够经过if...then...else...结构,让Bash脚本从两个代码块中选择一个执行。该选择结构一样有一个条件。若是条件成立,那么将执行then附属的代码块,不然执行else附属的代码块。下面的demo_if_else.bash脚本是一个小例子:

#!/bin/bash filename=$1
if [ -e $filename ] then
 echo "$filename exists"
else
 echo "$filename NOT exists"
fi

echo "The End"

 

if后面的“-e $filename”做为判断条件。若是条件成立,即文件存在,那么执行then部分的代码块。若是文件不存在,那么脚本将执行else语句中的echo命令。末尾的fi结束整个语法结构。脚本继续以顺序的方式执行剩余内容。运行脚本: 

$./demo_if_else.bash a.out

脚本会根据a.out是否存在,打印出不一样的内容。

 

咱们看到,在使用if...then...else...结构时,咱们能够实现两部分代码块的选择执行。而在then代码块和else代码块内部,咱们能够继续嵌套选择结构,从而实现更多个代码块的选择执行。好比脚本demo_nest.bash:

#!/bin/bash var=`whoami` echo "You are $var"

if [ $var = "root" ] then
 echo "You are my God."
else
 if [ $var = "vamei" ]  then
 echo "You are a happy user."
 else
 echo "You are the Others."
 fi
fi

 

在Bash下,咱们还能够用case语法来实现多程序块的选择执行。好比下面的脚本demo_case.bash: 

#!/bin/bash var=`whoami` echo "You are $var"

case $var in root) echo "You are God." ;; vamei) echo "You are a happy user." ;; *) echo "You are the Others." ;; esac

这个脚本和上面的demo_nest.bash功能彻底相同。能够看到case结构与if结构的区别。关键字case后面再也不是逻辑表达式,而是一个做为条件的文本。后面的代码块分为三个部分,都以文本标签)的形式开始,以;;结束。在case结构运行时,会逐个检查文本标签。当条件文本和文本标签能够对应上时,Bash就会执行隶属于该文本标签的代码块。若是是用户vamei执行该Bash脚本,那么条件文本和vamei标签对应上,脚本就会打印:

You are a happy user.

 

文本标签除了是一串具体的文本,还能够包含文本通配符。结构case中经常使用的通配符包括:

 

通配符 含义 文本标签例子 经过的条件文本
* 任意文本 *) Xyz, 123, …
? 任意一个字符 a?c) abc, axc, …
[] 范围内一个字符 [1-5][b-d]) 2b, 3d, …

上面的程序中最后一个文本标签是通配符*,即表示任意条件文本均可以触发此段代码块的运行。固然,前提是前面的几个文本标签都没有“截胡”。

 

循环结构

循环结构是编程语言中另外一种常见的语法结构。循环结构的功能是重复执行某一段代码,直到计算机的状态符合某一条件。在while语法中,Bash会循环执行隶属于while的代码块,直到逻辑表达式不成立。好比下面的demo_while.bash:

#!/bin/bash now=`date +'%Y%m%d%H%M'` deadline=`date --date='1 hour' +'%Y%m%d%H%M'` while [ $now -lt $deadline ] do
 date
 echo "not yet"
 sleep 10 now=`date +'%Y%m%d%H%M'` done

echo "now, deadline reached"

关键字do和done之间的代码是隶属于该循环结构的代码块。在while后面跟着条件,该条件决定了代码块是否重复执行下去。这个条件是用当前的时间与目标时间对比。若是当前时间小于目标时间,那么代码块就会重复执行下去。不然,Bash将跳出循环,继续执行后面的语句。

 

若是while的条件始终是真,那么循环会一直进行下去。下面的程序就是以无限循环的形式,不断播报时间: 

#!/bin/bash while true 
do
  date
  sleep 1
done

 

语法while的终止条件是一个逻辑判断。若是在循环过程当中改变逻辑判断的内容,那么咱们很难在程序执行以前预判循环进行的次数。正如咱们以前在demo_while.bash中看到的,咱们在循环进行过程当中改变着做为条件的逻辑表达式,不断地更新参与逻辑判断的当前时间。与while语法对应的是for循环。这种语法会在程序进行前肯定好循环进行的次数,好比demo_for.bash: 

#!/bin/bash for var in `ls log*` do
  rm $var done

 

在这个例子中,命令ls log*将返回全部以log开头的文件名。这些文件名之间由空格分隔。循环进行时,Bash会依次取出一个文件名,赋值给变量var,并执行do和done之间隶属于for结构的程序块。因为ls命令返回的内容在是肯定的,所以for循环进行的次数也会在一开始肯定下来。

 

在for语法中,咱们也可使用本身构建一个由空格分隔的文本。由空格区分出来的每一个子文本会在循环中赋值给变量。好比:

#!/bin/bash for user in vamei anna yutian do
 echo $user done

 

此外,for循环还能够和seq命令配合使用。命令seq用于生成一个等差的整数序列。命令后面能够跟3个参数,第一个参数表示整数序列的开始数字,第二个参数表示每次增长多少,最后一个参数表示序列的终点。所以,下面命令: 

$seq 1 2 10

将返回:

1 3 5 7 9

能够看到,seq返回的也是由空格分隔开的文本。所以,seq的返回结果也可用于for循环。

 

结合for循环和seq命令,咱们能够解一些有趣的数学问题。好比高斯求和,是要计算从1到100的全部整数的和。咱们能够用Bash解决: 

#!/bin/bash total=0

for number in `seq 1 1 100` do total=$(( $total + $number )) done

echo $total

 

 这个问题还能够用do while循环来求解:

#!/bin/bash total=0 number=1
while : do
  if [ $number -gt 100 ] then break fi total=$(( $total + $number )) number=$(($number + 1)) done

echo $total

这里break语句的做用是在知足条件时跳出循环。

 

若是想计算1到100全部不被3整数的和,则可使用continue语句,跳过全部被3整数的数:

#!/bin/bash total=0
for number in `seq 1 1 100` do
  if (( $number % 3 == 0 )) then continue fi total=$(( $total + $number )) done

echo $total

 

Bash与C语言

到了这里,咱们已经介绍完Bash语言的基本语法。Bash语言和C语言都是Linux下的经常使用语言。它们都能经过特定的语法来编写程序,而程序运行后都能实现某些功能。尽管在语法细节上存在差别,但两种语言都有如下语法:

  • 变量:在内存中储存数据
  • 循环结构:重复执行代码块
  • 选择结构:根据条件执行代码块
  • 函数:复用代码块

编程语言的做者在设计语言时,每每会借鉴已有编程语言的优势。这是编程语言之间类似性的一大缘由。程序员每每要掌握不止一套编程语言。类似的语法特征,会让程序员在学习新语言时感到亲切,从而促进语言的推广。

Bash和C的类似性,也来自于它们共同遵照的编程范式——面向过程编程。支持面向过程编程的语言,通常都会提供相似于函数的代码封装方式。函数把多行指令包装成一个功能。只要知道了函数名,程序能够经过调用函数来使用函数功能,最终实现代码复用。除了面向过程编程,还有面向对象和函数式的编程范式。每种编程范式都提供了特定的代码封装方式,并达到代码复用的目的。值得注意的是,近年来出现的新语言每每会支持不止一种编程范式。

除了类似性,咱们还应该注意到Bash和C程序的区别。Bash的变量只能是文本类型,C的变量却能够有整数、浮点数、字符等类型。Bash的不少功能,如加减乘除运算,都是调用其余程序实现的。而C直接就能够进行加减乘除运算。能够说,C语言是一门真正的编程语言。C程序最终会编译成二进制的可执行文件。CPU能够直接理解这些文件中的指令。

另外一方面,Bash是一个Shell。它本质上是一个命令解释器程序,而不是编程语言。用户能够经过命令行的方式,来调用该程序的某些功能。所谓的Bash编程,只是命令解释器程序提供的一种互动方法。Bash脚本只能和Bash进程互动。它不能像C语言同样,直接调用CPU的功能。所以,Bash能实现的功能会受限,运行速度上也比不上可执行文件。

但另外一反面,Bash脚本也有它的好处。 C语言能接触到很底层的东西,但使用起来也很复杂。有时候,即便你已经知道如何用C实现一个功能,写代码依然是一个很繁琐的过程。Bash正相反。因为Bash能够便捷地调用已有的程序,所以不少工做能够用数行的脚本解决。此外,Bash脚本不须要编辑,就能够由Bash进程理解并执行。所以,开发Bash脚本比写C程序要快不少。Linux的系统运维工做,如按期备份、文件系统管理等,就常用到Bash脚本。总之,Bash编程知识是晋级为资深Linux用户的必要条件。

 

欢迎阅读“骑着企鹅采树莓”系列文章

相关文章
相关标签/搜索