赞
踩
1.实验目的与要求
2.实验平台
实验室安装的实验环境(Linux操作系统)和头歌(www.educoder.net)实验平台(课程实验)
3.实验内容
4.实验详细内容、步骤
本关任务:欢迎来到shell
脚本世界,为了见证shell
脚本的神奇,本小节带领大家写第一个shell脚本,希望脚本能够在右侧的命令行窗口中输出hello EduCoder
。
接下来让我们一起去开启shell
脚本的快乐之旅吧!
Shell是一个用 C 语言编写的程序,它是用户使用 Linux
的桥梁。Shell
既是一种命令语言,又是一种程序设计语言。为了更好地学习shell
编程,您需要对linux
系统命令有一定的了解,熟悉常用的命令,如:vim
、echo、chmod
等。
shell 语言编写的程序通常都非常轻巧,比如我想编写一个在屏幕上显示Hello World !
程序:
说明:#!
是一个约定的标记,它告诉系统这个脚本需要什么解释器来执行,即使用哪一种 shell
。echo
命令用于向窗口输出文本内容。
Shell 脚本的运行通常有两种方法: 1、作为可执行程序运行
注意:一定要写成 ./test.sh
,而不是 test.sh
,./
的作用其实就是告诉系统就在当前目录查找,否则系统会去PATH
路径中查找有没有test.sh
的文件,而系统默认情况下只有 /bin
, /sbin
, /usr/bin
,/usr/sbin
等在 PATH
中,很难找到。
2、作为解释器参数运行,这种运行方式是直接运行解释器,其参数就是 shell 脚本的文件名,如:
这种方式运行的脚本,不需要在第一行指定解释器信息。
请按照要求,完成以下实践内容。 1、在/opt/
目录下创建第一个shell
脚本 文件命名为 test.sh
; 2、编写test.sh
脚本,让其输出Hello EduCoder
(通过 vim 编辑器编辑); 3、给/opt/test.sh
赋予执行权限; 4、运行test.sh
文件。
本关带领大家熟悉 shell 的变量并掌握其使用。
从字面上来看就是可以变的量,举个例子,我们小时候都做过数学的应用题,经常定义 y 的值是某个数,如果换了一道题,还是定义 y 的值,但是 y 的值就不和第一道题相同了,这个 y 就是变量。
变量是计算机内存的单元,在shell中变量其中存放的值可以改变。当 Shell 脚本需要保存一些信息时,如一段字符串或者一段数据,我们可以通过变量赋值的方式把他保存在变量中。
每个变量都有一个名字,所以命令变量,尽量做到,见名知义。 那么,在 shell 中应该如何设置变量呢?其实 so easy,命令如下:
运行 shell 时,会同时存在三种变量:
特殊变量:shell 对一些参数做特殊处理,这些参数只能被引用而不能被赋值。
1)在定义变量时,有一些规则需要遵守。
变量名可以由字母、数字和下画线组成,但是不能以数字开头,不能使用程序中的关键字(保留字),要见名知义。如果变量名是"20name或者if",则是错误的。
2)在 Bash 中,变量的默认类型都是字符串型,如果要进行数值运算,则必须指定变量类型为数值型。比如:
看到了吧,变量 aa 的值不是"3",而是"1+2"。在 Bash 中,变量类型是字符串型,所以认为"1+2"只是一个字符串,而不会进行数值运算(数值运算方法会在后续章节中介绍)。
3) 变量用等号"="连接值,"="左右两侧不能有空格。这是 Shell 语言特有的格式要求。在绝大多数的其他语言中,"="左右两侧是可以加入空格的。但是在 Shell 中命令的执行格式是"命令 [选项] [参数]",如果在"="左右两侧加入空格,那么 Linux 会误以为这是系统命令,是会报错的。
4) 变量值中如果有空格,则需要使用单引号或双引号包含,如 test="hello world!"。双引号括起来的内容"$"和反引号者都拥有特殊含义,而单引号括起来的内容都是普通字符。
5) 在变量值中,可以使用转义符"\"
。
6) 如果需要増加变量值,那么可以进行变量叠加。 例如:
7) 如果要把命令的执行结果作为变量值赋予变量,则需要使用反引号或 $() 包含命令。例如:
8) 环境变量名建议大写,便于区分。
使用 unset 命令可以删除变量。语法:
变量被删除后不能再次使用。unset 命令不能删除只读变量(readonly myUrl 关键字 readonly 定义的变量叫做只读变量),如下:
/opt/test2.sh
, 已知向 /opt/test.sh
传递参数 "www" "educoder" "net" 三个字符串。/opt/test.sh
需要实现如下功能:输出传递的参数的个数,并且输出第一个参数。比如: 向 /opt/test2.sh
传递 是"a" "b" "c" "d",则执行 bash /opt/test.sh a b c d
输出的结果应该是 4 a
。
任务描述
本关带领大家熟悉 shell 的变量并掌握其使用。
字符串是 shell 编程中最常用最有用的数据类型(除了数字和字符串,也没啥其它类型好用了),字符串可以用单引号,也可以用双引号,也可以不用引号。
双引号 中可以有变量; 双引号里可以先转义字符;双引号中的单引号输出时维持单引号不变。请下面的案例: 例 1:
例 2:
单引号: 会忽略所有的特殊字符,即任何字符都会原样输出,包括定义的变量; 单引号字串中不能出现单引号。
反引号 : 有命令替换的作用见例 3;反引号可以嵌套使用,但内层的单引号必须加上\
进行转义见例 4。 例 3:
例 4:
示例:
获取字符串长度
字符串截取
字符串匹配删除
字符串替换
字符串比较 四种判断方式 == != < >
(按 ascii 值比较大小), ==
相等则为真,!=
不相等则为真。
按照右侧“代码文件”中提示的要求补全代码,本关的目的,是希望通过学习,对于字符串的使用能有一个大致的了解,在实际工作中遇到需要对某一个字符串处理的时候,能够找到对应的方法。
任务描述
本关将带领大家熟悉 bash 的内置命令,并掌握一些常用的内置命令。 在开启此小节之前,相信你已经对 shell 编程的变量和字符串的使用有了一定的了解。需注意: 本课程没有描述 shell 数组的概念,因为在实际的操作中 shell 数组并不是很友好,一般在数据处理的模型都用 python 来代替 shell,如果大家对 shell 数组感兴趣,请同学们可以自行翻阅资料了解一下 shell 的数组概念。
所谓 Shell 内建命令,就是由 Bash 自身提供的命令,而不是文件系统中的某个可执行文件。 例如,用于进入或者切换目录的 cd 命令,虽然我们一直在使用它,但如果不加以注意,很难意识到它与普通命令的性质是不一样的:该命令并不是某个外部文件,只要在 Shell 中你就可以运行这个命令。 可以使用 type
来确定一个命令是否是内建命令:
由此可见,cd 是一个 Shell 内建命令,而 ifconfig 是一个外部文件,它的位置是/sbin/ifconfig
。 还记得系统变量 $PATH
吗?$PATH
变量包含的目录中几乎聚集了系统中绝大多数的可执行命令,它们都是外部命令。
通常来说,内建命令会比外部命令执行得更快,执行外部命令时不但会触发磁盘 I/O,还需要 fork 出一个单独的进程来执行,执行完成后再退出。而执行内建命令相当于调用当前 Shell 进程的一个函数。
下表列出了 Bash Shell 中直接可用的内建命令。
命令 说明
一下子掌握所有的内建命令难度要求比较大,我们可以从常用内建命令入手,第一个内置命令: alias
。 1、alias alias 用来给命令创建一个别名。若直接输入该命令且不带任何参数,则列出当前 Shell 进程中使用了哪些别名,如下:
现在你应该能理解类似ll
这样的命令为什么与ls -l
或者 ls -alF
的效果是一样的吧,是因为ls -alF
被重新命名为ll
。因此,在命令行输入 ll
,即相当于输入了 ls -alF
。
使用 alias 命令自定义别名的语法格式为(临时):
通过 date 命令可以获得当前的 UNIX 时间戳,具体写法为 date +%s
,如果你嫌弃它太长或者不容易记住,那可以给它定义一个别名。
这样你在命令行输入 timestamp
获得的效果及跟输入date +%s
一样,如下:
注意:通过以上方式命令别名后,在退出当前窗口后,刚设定的 alias 命令别名也就消失了,如何才能永久使用了?请看下面的介绍:
首先,如果想让别名能够在不同的 shell 会话中使用,就需要把它们添加到 shell 用来保存别名的文件中。大多数情况下,这个文件不是 .bashrc
,就是 .bash_aliases
,我们使用的系统 CentOS,是保存在.bashrc
里。(或者可以在 root 家目录下输入:ls -a ~
看到那个文件)
如果要永久保存,就把alias new_name='command'
给编辑到 /root/.bashrc
下,然后:wq
保存退出,但得重新加载别名文件,才能生效,可执行 source ~/.bashrc
(这样更改后只有 root 用户的别名生效,要想普通用户有效,普通用户目录下的.bashrc
也必需作同样的修改!比如现在有用户为 jerry,则需要到其/home/jerry/.bashrc
下编辑然后执行source ~/.bashrc
)。
实际用例:假设生产环境有三台机器127.1.1.2
;127.1.1.3
;127.1.1.4
; 你可以通过 alias 的方式简化 ssh 登录方式,操作如下: 在~/.bashrc
下末尾行追加内容。然后 source ~/.bashrc
,这样即可以在命令行直接输入 ssh_01
,ssh_02,ssh_03
快速登陆。
本关没有评测,希望大家独立在命令行按照示例自己操作一遍,了解熟悉 shell 的内置命令,并掌握 alias 的使用方式以及应用场景。
本关继续为大家介绍 shell 的内置命令 echo 和 read,通过学习掌握 echo 和 read 的用法。
echo 是一个 Shell 内建命令,用来在终端输出字符串,并在最后默认加上换行符。请看下面的例子:
输出结果为:
echo 命令输出结束后默认会换行,如果不希望换行,可以加上-n
参数,如下所示:
输出结果为:
默认情况下,echo 不会解析以反斜杠\
开头的转义字符。比如,\n
表示换行,echo 默认会将它作为普通字符对待。请看下面的例子:
我们可以添加-e
参数来让 echo 命令解析转义字符。例如:
read 是 Shell 内置命令,用来从标准输入中读取数据并赋值给变量。如果没有进行重定向,默认就是从键盘读取用户输入的数据;如果进行了重定向,那么可以从文件中读取数据(后续我们会在 shell 的输入和输出中深入讲解重定向的概念,不了解的同学可以不用理会,暂时就认为:read 命令就是从键盘读取数据)。
read 命令的用法为:
options 表示选项,如下表所示;variables 表示用来存储数据的变量,可以有一个,也可以有多个。
options 和 variables 都是可选的,如果没有提供变量名,那么读取的数据将存放到环境变量 REPLY 中。 Shell read 命令支持的选项:
【实例1】使用 read 命令给多个变量赋值,并打印提示语 read -p
。
输出结果为:
Tips: 如果用 read 交互参数,输入错了,请按住Ctrl+Backspace清除而不是只按住Backspace。
【示例2】只读取一个字符。
运行结果:
【实例3】在指定时间内输入密码。
这段代码中,我们使用**&&**组合了多个命令,相当于其他语言中的 and ,这些命令会依次执行,并且从整体上作为 if 语句的判断条件。只要其中一个命令执行失败(退出状态为非 0 值),整个判断条件就失败了,后续的命令也就没有必要执行了。
关于 echo 和 read 的使用方式还有很多,大家可以通过 help echo 和 help read 的方式去获取更多的帮助文档。
本关继续带领大家了解掌握 shell 的内置命令,接下来将会学习内置命令exit
和declare
。
exit 是一个 Shell 内置命令,用来退出当前 Shell 进程,并返回一个退出状态;使用**$?**可以接收这个退出状态,$?
如果不记得是什么,请查看前一章节的shell变量。exit
命令可以接受一个整数值作为参数,代表退出状态。如果不指定,默认状态值是 0。
一般情况下,退出状态为 0 表示成功,退出状态为非 0 表示执行失败(出错)了, exit
退出状态只能是一个介于 0~255 之间的整数,其中只有 0 表示成功,其它值都表示失败。
Shell 进程执行出错时,可以根据退出状态来判断具体出现了什么错误,比如打开一个文件时,我们可以指定 1 表示文件不存在,2 表示文件没有读取权限,3 表示文件类型不对。 【实列1】运行该脚本:
可以看到,"after exit"并没有输出,这说明遇到 exit 命令后,test.sh 执行就结束了。 注意:exit 表示退出当前 Shell 进程,我们必须在新进程中运行 test.sh,否则当前 Shell 会话(终端窗口)会被关闭,我们就无法看到输出结果了。 我们可以紧接着使用$?
来获取 test.sh 的退出状态:
declare
一般很少用到,但大家要知道下列常用的方法。
【实列2】declare 将变量声明为整数并进行计算。
如果我们不使用 declare,结果如下所示:
因此注意在 shell 如果要求整数值的话,不要忘了有 declare 这样的内置命令来定义变量。关于 declare 的其他使用方式,大家可以自行去了解。
补全右侧编辑器 Begin-End 区间的代码,完成指定功能,具体要求如下:
1.现有变量 a、b、c ,已知 a 为整数 100,b 为整数 300,c=$a+$b
,打印 c 的值,应该为 400; 2.补全[ $? -eq ]内的值,-eq 的意思是‘等于’,注意 [ ] 两边要留一个空格,通过 $? 判断上面的命令是否运行成功,如果运行成功则输出success
,否则输出 faild
。
注意:请不要直接输出 400 或者直接输出 success
,因为评测程序会判断你的代码文件。
任务描述
本关任务:掌握函数的创建和使用方法。
函数是一个脚本代码块,你可以为其命名并在代码中任何位置重用。要在脚本中使用该代码块时,只要使用所起的函数名就行了(这个过程称为调用函数)。本关将会介绍如何在 shell 脚本中创建和使用函数。
有两种格式可以用来在 bash shell 脚本中创建函数。
参数说明:
函数名后的空括号表明正在定义的是一个函数。这种格式的命名规则和之前定义 shell 脚本函数的格式一样。
【实例1】要在脚本中使用函数,只需要像其他 shell 命令一样,在行中指定函数名就行了。 已知 cat test.sh
显示内容如下:
执行输出结果如下:
每次引用函数名 func1 时,bash shell 会找到 func1 函数的定义并执行你在那里定义的命令。 【实例2】函数的调用要在函数定义之后使用,否则会报错。
执行结果如下:
【实例3】函数名必须是唯一的,否则也会有问题。如果你重定义了函数,新定义会覆盖原来函数的定义,这一切不会产生任何错误消息。
执行结果如下:
在右侧编辑器 Begin-End 区间补充代码,调用 welcome 函数输出指定内容。
任务描述
本关任务:学习掌握什么是 shell 函数的返回值。
bash shell 会把函数当作一个小型脚本,运行结束时,会返回一个退出状态码。有 3 种不同的方法来为函数生成退出状态码。
默认情况下,函数的退出状态码是函数中最后一条命令返回的退出状态码。在函数执行结束后,可以用标准变量$?
来确定函数的退出状态码。 【实例1】已知 test1.sh 脚本内容如下:
运行脚本 bash test1.sh
执行结果如下:
函数的退出状态码是 2,这是因为函数中的最后一条命令没有成功运行。但你无法知道函数中其他命令中是否成功运行。带着这个疑问再看下面的例子:
【实例2】已知 test2.sh 脚本内容如下:
运行脚本 bash test1.sh
执行结果如下:
这次,由于函数最后一条语句 echo 运行成功,该函数的退出状态码就是 0,尽管其中有一条命令并没有正常运行。由此可见,使用函数的默认退出状态码,来判断函数是否正常运行是很危险的。幸运的是,有几种办法可以解决这个问题。
bash shell 使用 return 命令退出函数,并返回特定的退出状态码。return 命令允许指定一个整数值来定义函数的退出状态码,从而提供了一种简单的途径来设定函数退出状态码。 【实例3】已知 test3.sh 脚本的内容如下:
执行bash test.sh
显示如下:
这一次变量$?
返回的结果是 20,而按照默认的返回 $?
应该是 0,因为函数的最后一条命令正确执行。
本关任务:掌握函数传递参数的方法以及如何处理变量。
我们在 shell 变量那个实训,介绍过了一些特殊变量 $0
、$1
、$#
等。在 shell 中,函数可以使用标准的参数环境变量来表示命令行上传给函数的参数。例如,函数名会在$0
变量中定义,函数命令行上的任何参数都会通过$1
、$2
等定义。也可以用特殊变量$#
,来判断传给函数的参数数目。请参考实例 1:
【实例1】已知脚本 test.sh 的内容如下:
输出结果如下:
这段脚本的意思是:脚本中的 addem 函数首先会检查脚本传给它的参数数目。如果没有任何参数,或者参数多于两个,addem 会返回值 -1;如果只有一个参数,addem 会将参数与自身相加;如果有两个参数,addem 会将它们进行相加。
注意:由于函数使用特殊参数环境变量作为自己的参数值,因此它无法直接获取脚本在命令行中的参数值。下面的【实例2】将会运行失败。
【实例2】:已知 test2.sh 的内容如下:
我在命令行执行 bash test2.sh 10 3
,发现脚本报错:
尽管函数 badfunc1 也使用了 $1 和 $2 变量,但它们和脚本主体中的 $1 和 $2 变量并不相同。要在函数中使用这些值,必须在调用函数时手动将它们传过去。我们把脚本改动如下:
通过将$1
和$2
变量传给函数,它们就能跟其他变量一样供函数使用了,我在命令行执行 bash test2.sh 10 3
,结果符合预期:
函数使用两种类型的变量:
全局变量是在 shell 脚本中任何地方都有效的变量。如果你在脚本的主体部分定义了一个全局变量,那么可以在函数内读取它的值。类似地,如果你在函数内定义了一个全局变量,可以在脚本的主体部分读取它的值。 默认情况下,你在脚本中定义的任何变量都是全局变量。在函数外定义的变量可在函数内正常访问,请看【实例3】。
【实列3】查看 test3.sh 内容,执行cat test3.sh
得到如下内容:
执行脚本输出结果如下:
$value
变量在函数外定义并被赋值。当 dbl 函数被调用时,该变量及其值在函数中都依然有效。如果变量在函数内被赋予了新值,那么在脚本中引用该变量时,新值也依然有效。但这其实很危险,尤其是如果你想在不同的 shell 脚本中使用函数的话。它要求你清清楚楚地知道函数中具体使用了哪些变量,包括那些用来计算非返回值的变量。请看【实例4】。
【实例4】:
执行脚本输出结果如下:
无需在函数中使用全局变量,函数内部使用的任何变量都可以被声明成局部变量。要实现这一点,只要在变量声明的前面加上 local 关键字就可以了,比如local temp
。或者也可以在变量赋值语句中使用 local 关键字:local temp=$[ $value + 5 ]
。 local 关键字保证了变量只局限在该函数中。如果脚本中在该函数之外有同样名字的变量,那么 shell 将会保持这两个变量的值是分离的。现在你就能很轻松地将函数变量和脚本变量隔离开了,只共享需要共享的变量。
【实例5】:
执行脚本输出结果如下:
现在,在 func1 函数中使用 $temp
变量时,并不会影响在脚本主体中赋给$temp
变量的值。请重点分析【实例4】和【实例5】中的不同。
本关不进行考核,希望大家能够真正的理解:
本关任务:掌握文件包含方法的使用。
在 C,C++,PHP 中都是用 include 来包含文件,Go 和 Java 使用 import 来包含(导入)包,而在 shell 中,很简单,只需要一个点.
,然后跟着文件路径及文件名,或者使用source
关键字也可以,注意文件路径可以使用绝对路径和相对路径。请看实例 1。
【实例1】已知在同一目录下,有 test1.sh 和 test2.sh 两个脚本,test1.sh 内容如下:
同一目录下的 test2.sh 内容如下:
在命令行执行bash test2.sh
,大家想想看这个时候输出的是什么? 输出的结果如下:
注意:
【实例2】同一目录下 test1.sh、test2.sh、test3.sh 内容如下:
在命令行执行bash test3.sh
,输出结果如下:
.
的方式包含进来,通过函数名 hello 的调用,调用了 test2.sh 中的 hello()函数,输出“i am from test3.sh” 。本关任务包含了三个脚本文件,它们分别是 step4/t1.sh
、step4/t2.sh
、 step4/main.sh
,点击右侧“代码文件”即可查看,如下图所示:
step4/t1.sh
代码。step4/main.sh
代码。Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。