BASH的基本语法
- 最简单的例子—— Hello World!
- 关于输入、输出和错误输出
- BASH中对变量的规定(与C语言的异同)
- BASH中的基本流程控制语法
- 函数的使用
2.1 最简单的例子—— Hello World!
几乎所有的讲解编程的书给读者的第一个例子都是Hello World程序,那么我们今天也就从这个例子出发,来逐步了解BASH。
用vi编辑器编辑一个hello文件如下:
#!/bin/bash
# This is a very simple example
echo Hello World
这样最简单的一个BASH程序就编写完了。这里有几个问题需要说明一下:
一,第一行的#!是什么意思
二,第一行的/bin/bash又是什么意思
三,第二行是注释吗
四,echo语句
五,如何执行该程序
#!是说明hello这个文件的类型的,有点类似于Windows系统下用不同文件后缀来表示不同文件类型的意思(但不相同)。Linux系统根据"#!"及该字串后面的信息确定该文件的类型,关于这一问题同学们回去以后可以通过"man magic"命令及/usr/share/magic文件来了解这方面的更多内容。在BASH中第一行的"#!"及后面的"/bin/bash"就表明该文件是一个BASH程序,需要由/bin目录下的bash程序来解释执行。BASH这个程序一般是存放在/bin目录下,如果你的Linux系统比较特别,bash也有可能被存放在/sbin、/usr/local/bin、/usr/bin、/usr/sbin或/usr/local/sbin这样的目录下;如果还找不到,你可以用"locate bash" "find / -name bash 2> /dev/null"或"whereis bash"这三个命令找出bash所在的位置;如果仍然找不到,那你可能需要自己动手安装一个BASH软件包了。
第二行的"# This is a ..."就是BASH程序的注释,在BASH程序中从"#"号(注意:后面紧接着是"!"号的除外)开始到行尾的多有部分均被看作是程序的注释。的三行的echo语句的功能是把echo后面的字符串输出到标准输出中去。由于echo后跟的是"Hello World"这个字符串,因此"Hello World"这个字串就被显示在控制台终端的屏幕上了。需要注意的是BASH中的绝大多数语句结尾处都没有分号。
如何执行该程序呢?有两种方法:一种是显式制定BASH去执行:
$ bash hello或
$ sh hello(这里sh是指向bash的一个链接,"lrwxrwxrwx 1 root root 4 Aug 20 05:41 /bin/sh -> bash")
或者可以先将hello文件改为可以执行的文件,然后直接运行它,此时由于hello文件第一行的"#! /bin/bash"的作用,系统会自动用/bin/bash程序去解释执行hello文件的:
$ chmod u+x hello
$ ./hello
此处没有直接"$ hello"是因为当前目录不是当前用户可执行文件的默认目录,而将当前目录"."设为默认目录是一个不安全的设置。
需要注意的是,BASH程序被执行后,实际上Linux系统是另外开设了一个进程来运行的。
2.2 关于输入、输出和错误输出
在字符终端环境中,标准输入/标准输出的概念很好理解。输入即指对一个应用程序或命令的输入,无论是从键盘输入还是从别的文件输入;输出即指应用程序或命令产生的一些信息;与Windows系统下不同的是,Linux系统下还有一个标准错误输出的概念,这个概念主要是为程序调试和系统维护目的而设置的,错误输出于标准输出分开可以让一些高级的错误信息不干扰正常的输出信息,从而方便一般用户的使用。
在Linux系统中:标准输入(stdin)默认为键盘输入;标准输出(stdout)默认为屏幕输出;标准错误输出(stderr)默认也是输出到屏幕(上面的std表示standard)。在BASH中使用这些概念时一般将标准输出表示为1,将标准错误输出表示为2。下面我们举例来说明如何使用他们,特别是标准输出和标准错误输出。
输入、输出及标准错误输出主要用于I/O的重定向,就是说需要改变他们的默认设置。先看这个例子:
$ ls > ls_result
$ ls -l >> ls_result
上面这两个命令分别将ls命令的结果输出重定向到ls_result文件中和追加到ls_result文件中,而不是输出到屏幕上。">"就是输出(标准输出和标准错误输出)重定向的代表符号,连续两个">"符号,即">>"则表示不清除原来的而追加输出。下面再来看一个稍微复杂的例子:
$ find /home -name lost* 2> err_result
这个命令在">"符号之前多了一个"2","2>"表示将标准错误输出重定向。由于/home目录下有些目录由于权限限制不能访问,因此会产生一些标准错误输出被存放在err_result文件中。大家可以设想一下find /home -name lost* 2>>err_result 命令会产生什么结果?
如果直接执行 find /home -name lost* > all_result ,其结果是只有标准输出被存入all_result文件中,要想让标准错误输出和标准输入一样都被存入到文件中,那该怎么办呢?看下面这个例子:
$ find /home -name lost* > all_result 2>& 1
上面这个例子中将首先将标准错误输出也重定向到标准输出中,再将标准输出重定向到all_result这个文件中。这样我们就可以将所有的输出都存储到文件中了。为实现上述功能,还有一种简便的写法如下:
$ find /home -name lost* >& all_result
如果那些出错信息并不重要,下面这个命令可以让你避开众多无用出错信息的干扰:
$ find /home -name lost* 2> /dev/null
同学们回去后还可以再试验一下如下几种重定向方式,看看会出什么结果,为什么?
$ find /home -name lost* > all_result 1>& 2
$ find /home -name lost* 2> all_result 1>& 2
$ find /home -name lost* 2>& 1 > all_result
另外一个非常有用的重定向操作符是 "-",请看下面这个例子:
$ (cd /source/directory && tar cf - . ) | (cd /dest/directory && tar xvfp -)
该命令表示把/source/directory目录下的所有文件通过压缩和解压,快速的全部移动到/dest/directory目录下去,这个命令在/source/directory和/dest/directory不处在同一个文件系统下时将显示出特别的优势。
下面还几种不常见的用法:
n<&-表示将n号输入关闭
<&-表示关闭标准输入(键盘)
n>&-表示将n号输出关闭
>&-表示将标准输出关闭
2.3 BASH中对变量的规定(与C语言的异同)
好了下面我们进入正题,先看看BASH中的变量是如何定义和使用的。对于熟悉C语言的程序员,我们将解释BASH中的定义和用法与C语言中有何不同。
2.3.1. BASH中的变量介绍
我们先来从整体上把握一下BASH中变量的用法,然后再去分析BASH中变量使用与C语言中的不同。BASH中的变量都是不能含有保留字,不能含有"-"等保留字符,也不能含有空格。
2.3.1.1简单变量
在BASH中变量定义是不需要的,没有"int i"这样的定义过程。如果想用一个变量,只要他没有在前面被定义过,就直接可以用,当然你使用该变量的第一条语句应该是对他赋初值了,如果你不赋初值也没关系,只不过该变量是空(注意:是NULL,不是0)。不给变量赋初值虽然语法上不反对,但不是一个好的编程习惯。好了我们看看下面的例子:
首先用vi编辑下面这个文件hello2:
#!/bin/bash
# give the initialize value to STR
STR="Hello World"
echo $STR
在上面这个程序中我们需要注意下面几点:
一,变量赋值时,'='左右两边都不能有空格;
二,BASH中的语句结尾不需要分号(";");
三,除了在变量赋值和在FOR循环语句头中,BASH中的变量使用必须在变量前加"$"符号,同学们可以将上面程序中第三行改为"echo STR"再试试,看看会出什么结果。==>output: STR
四,由于BASH程序是在一个新的进程中运行的,所以该程序中的变量定义和赋值不会改变其他进程或原始Shell中同名变量的值,也不会影响他们的运行。
更细致的文档甚至提到以但引号括起来的变量将不被BASH解释为变量,如 '$STR' ,而被看成为纯粹的字符串。而且更为标准的变量引用方式是${STR}这样的,$STR自不过是对${STR}的一种简化。在复杂情况下(即有可能产生歧义的地方)最好用带{}的表示方式。
BASH中的变量既然不需要定义,也就没有类型一说,一个变量即可以被定义为一个字符串,也可以被再定义为整数。如果对该变量进行整数运算,他就被解释为整数;如果对他进行字符串操作,他就被看作为一个字符串。请看下面的例子:
#!/bin/bash
x=1999
let "x = $x + 1"
echo $x
x="olympic'"$x
echo $x
关于整数变量计算,有如下几种:" + - * / % ",他们的意思和字面意思相同。整数运算一般通过let和expr这两个指令来实现,如对变量x加1可以写作:let "x = $x + 1" 或者 x=`expr $x + 1`
在比较操作上,整数变量和字符串变量各不相同,详见下表:
对应的操作 | 整数操作 | 字符串操作 |
相同 | -eq | = |
不同 | -ne | != |
大于 | -gt | > |
小于 | -lt | < |
大于或等于 | -ge |
|
小于或等于 | -le |
|
为空 |
| -z |
不为空 |
| -n |
比如:
比较字符串a和b是否相等就写作:if [ $a = $b ]
判断字符串a是否为空就写作: if [ -z $a ]
判断整数变量a是否大于b就写作:if [ $a -gt $b ]
更细致的文档推荐在字符串比较时尽量不要使用-n,而用! -z来代替。(其中符号"!"表示求反操作)
BASH中的变量除了用于对整数和字符串进行操作以外,另一个作用是作为文件变量。BASH是Linux操作系统的Shell,因此系统的文件必然是BASH需要操作的重要对象,如 if [ -x /root ]可以用于判断/root目录是否可以被当前用户进入。下表列出了BASH中用于判断文件属性的操作符:
运算符 | 含义(满足下面要求时返回TRUE) |
-e file | 文件file已经存在 |
-f file | 文件file是普通文件 |
-s file | 文件file大小不为零 |
-d file | 文件file是一个目录 |
-r file | 文件file对当前用户可以读取 |
-w file | 文件file对当前用户可以写入 |
-x file | 文件file对当前用户可以执行 |
-g file | 文件file的GID标志被设置 |
-u file | 文件file的UID标志被设置 |
-O file | 文件file是属于当前用户的 |
-G file | 文件file的组ID和当前用户相同 |
file1 -nt file2 | 文件file1比file2更新 |
file1 -ot file2 | 文件file1比file2更老 |
注意:上表中的file及file1、file2都是指某个文件或目录的路径。
2.3.1.1.关于局部变量
在BASH程序中如果一个变量被使用了,那么直到该程序的结尾,该变量都一直有效。为了使得某个变量存在于一个局部程序块中,就引入了局部变量的概念。BASH中,在变量首次被赋初值时加上local关键字就可以声明一个局部变量,如下面这个例子:
#!/bin/bash
HELLO=Hello
function hello {
local HELLO=World
echo $HELLO
}
echo $HELLO
hello
echo $HELLO
该程序的执行结果是:
Hello
World
Hello
这个执行结果表明全局变量$HELLO的值在执行函数hello时并没有被改变。也就是说局部变量 $HELLO的影响只存在于函数那个程序块中。
2.3.2. BASH中的变量与C语言中变量的区别
这里我们为原来不熟悉BASH编程,但是非常熟悉C语言的程序员总结一下在BASH环境中使用变量需要注意的问题。
1,BASH中的变量在引用时都需要在变量前加上"$"符号(第一次赋值及在For循环的头部不用加"$"符号);
2,BASH中没有浮点运算,因此也就没有浮点类型的变量可用;
3,BASH中的整形变量的比较符号与C语言中完全不同,而且整形变量的算术运算也需要经过let或expr语句来处理;
2.4 BASH中的基本流程控制语法
BASH中几乎含有C语言中常用的所有控制结构,如条件分支、循环等,下面逐一介绍。
2.4.1 if...then...else
if语句用于判断和分支,其语法规则和C语言的if非常相似。其几种基本结构为:
if [ expression ]
then
statments
fi
DedeCms 会员中心注入漏洞-20130715 linux下IPTABLES配置详解