一、 Shell简介:什么是Shell,Shell命令的两种执行方式1php
二、 几种常见的Shell1html
6、Shell变量:Shell变量的定义、删除变量、只读变量、变量类型5bash
7、Shell特殊变量:Shell $0, $#, $*, $@, $?, $$和命令行参数6编程语言
八、 Shell替换:Shell变量替换,命令替换,转义字符9编辑器
九、 Shell运算符:Shell算数运算符、关系运算符、布尔运算符、字符串运算符12ide
二11、Shell函数:Shell函数返回值、删除函数、在终端调用函数35
二十3、Shell输入输出重定向:Shell Here Document,/dev/null文件38
与其余编程语言相似,Shell支持for循环。
for循环通常格式为:
for 变量 in 列表do command1 command2 ... commandNdone
列表是一组值(数字、字符串等)组成的序列,每一个值经过空格分隔。每循环一次,就将列表中的下一个值赋给变量。
in 列表是可选的,若是不用它,for 循环使用命令行的位置参数。
例如,顺序输出当前列表中的数字:
for loop in 1 2 3 4 5
do
echo "The value is: $loop"
done
运行结果:
The value is: 1The value is: 2The value is: 3The value is: 4The value is: 5
顺序输出字符串中的字符:
for str in 'This is a string'
do
echo $str
done
运行结果:
This is a string
显示主目录下以 .bash 开头的文件:
#!/bin/bash
for FILE in $HOME/.bash*
do
echo $FILE
done
运行结果:
/root/.bash_history/root/.bash_logout/root/.bash_profile/root/.bashrc
while循环用于不断执行一系列命令,也用于从输入文件中读取数据;命令一般为测试条件。其格式为:
while commanddo Statement(s) to be executed if command is truedone
命令执行完毕,控制返回循环顶部,从头开始直至测试条件为假。
如下是一个基本的while循环,测试条件是:若是COUNTER小于5,那么返回 true。COUNTER从0开始,每次循环处理时,COUNTER加1。运行上述脚本,返回数字1到5,而后终止。
COUNTER=0
while [ $COUNTER -lt 5 ]
do
COUNTER='expr $COUNTER+1'
echo $COUNTER
done
运行脚本,输出:
12345
while循环可用于读取键盘信息。下面的例子中,输入信息被设置为变量FILM,按<Ctrl-D>结束循环。
echo 'type <CTRL-D> to terminate'
echo -n 'enter your most liked film: '
while read FILM
do
echo "Yeah! great film the $FILM"
done
运行脚本,输出相似下面:
type <CTRL-D> to terminateenter your most liked film: Sound of MusicYeah! great film the Sound of Music
until 循环执行一系列命令直至条件为 true 时中止。until 循环与 while 循环在处理方式上恰好相反。通常while循环优于until循环,但在某些时候,也只是极少数状况下,until 循环更加有用。
until 循环格式为:
until commanddo Statement(s) to be executed until command is truedone
command 通常为条件表达式,若是返回值为 false,则继续执行循环体内的语句,不然跳出循环。
例如,使用 until 命令输出 0 ~ 9 的数字:
#!/bin/bash
a=0
until [ ! $a -lt 10 ]
do
echo $a
a=`expr $a + 1`
done
运行结果:
0123456789
在循环过程当中,有时候须要在未达到循环结束条件时强制跳出循环,像大多数编程语言同样,Shell也使用 break 和 continue 来跳出循环。
break命令
break命令容许跳出全部循环(终止执行后面的全部循环)。
下面的例子中,脚本进入死循环直至用户输入数字大于5。要跳出这个循环,返回到shell提示符下,就要使用break命令。
#!/bin/bash
while :
do
echo -n "Input a number between 1 to 5: "
read aNum
case $aNum in
1|2|3|4|5) echo "Your number is $aNum!"
;;
*) echo "You do not select a number between 1 to 5, game is over!"
break
;;
esac
done
在嵌套循环中,break 命令后面还能够跟一个整数,表示跳出第几层循环。例如:
break n
表示跳出第 n 层循环。
下面是一个嵌套循环的例子,若是 var1 等于 2,而且 var2 等于 0,就跳出循环:
#!/bin/bash
for var1 in 1 2 3
do
for var2 in 0 5
do
if [ $var1 -eq 2 -a $var2 -eq 0 ]
then
break 2
else
echo "$var1 $var2"
fi
done
done
如上,break 2 表示直接跳出外层循环。运行结果:
1 01 5
continue命令
continue命令与break命令相似,只有一点差异,它不会跳出全部循环,仅仅跳出当前循环。
对上面的例子进行修改:
#!/bin/bash
while :
do
echo -n "Input a number between 1 to 5: "
read aNum
case $aNum in
1|2|3|4|5) echo "Your number is $aNum!"
;;
*) echo "You do not select a number between 1 to 5!"
continue
echo "Game is over!"
;;
esac
done
运行代码发现,当输入大于5的数字时,该例中的循环不会结束,语句
echo "Game is over!"
永远不会被执行。
一样,continue 后面也能够跟一个数字,表示跳出第几层循环。
再看一个 continue 的例子:
#!/bin/bash
NUMS="1 2 3 4 5 6 7"
for NUM in $NUMS
do
Q=`expr $NUM % 2`
if [ $Q -eq 0 ]
then
echo "Number is an even number!!"
continue
fi
echo "Found odd number"
done
运行结果:
Found odd numberNumber is an even number!!Found odd numberNumber is an even number!!Found odd numberNumber is an even number!!Found odd number
二11、Shell函数:Shell函数返回值、删除函数、在终端调用函数
函数可让咱们将一个复杂功能划分红若干模块,让程序结构更加清晰,代码重复利用率更高。像其余编程语言同样,Shell 也支持函数。Shell 函数必须先定义后使用。
Shell 函数的定义格式以下:
function_name () { list of commands [ return value ]}
若是你愿意,也能够在函数名前加上关键字 function:
function function_name () { list of commands [ return value ]}
函数返回值,能够显式增长return语句;若是不加,会将最后一条命令运行结果做为返回值。
Shell 函数返回值只能是整数,通常用来表示函数执行成功与否,0表示成功,其余值表示失败。若是 return 其余数据,好比一个字符串,每每会获得错误提示:“numeric argument required”。
若是必定要让函数返回字符串,那么能够先定义一个变量,用来接收函数的计算结果,脚本在须要的时候访问这个变量来得到函数返回值。
先来看一个例子:
#!/bin/bash
# Define your function here
Hello () {
echo "Url is http://see.xidian.edu.cn/cpp/shell/"
}
# Invoke your function
Hello
运行结果:
$./test.shHello World$
调用函数只须要给出函数名,不须要加括号。
再来看一个带有return语句的函数:
#!/bin/bash
funWithReturn(){
echo "The function is to get the sum of two numbers..."
echo -n "Input first number: "
read aNum
echo -n "Input another number: "
read anotherNum
echo "The two numbers are $aNum and $anotherNum !"
return $(($aNum+$anotherNum))
}
funWithReturn
# Capture value returnd by last command
ret=$?
echo "The sum of two numbers is $ret !"
运行结果:
The function is to get the sum of two numbers...Input first number: 25Input another number: 50The two numbers are 25 and 50 !The sum of two numbers is 75 !
函数返回值在调用该函数后经过 $? 来得到。
再来看一个函数嵌套的例子:
#!/bin/bash
# Calling one function from another
number_one () {
echo "Url_1 is http://see.xidian.edu.cn/cpp/shell/"
number_two
}
number_two () {
echo "Url_2 is http://see.xidian.edu.cn/cpp/u/xitong/"
}
number_one
运行结果:
Url_1 is http://see.xidian.edu.cn/cpp/shell/Url_2 is http://see.xidian.edu.cn/cpp/u/xitong/
像删除变量同样,删除函数也可使用 unset 命令,不过要加上 .f 选项,以下所示:
$unset .f function_name
若是你但愿直接从终端调用函数,能够将函数定义在主目录下的 .profile 文件,这样每次登陆后,在命令提示符后面输入函数名字就能够当即调用。
在Shell中,调用函数时能够向其传递参数。在函数体内部,经过 $n 的形式来获取参数的值,例如,$1表示第一个参数,$2表示第二个参数...
带参数的函数示例:
#!/bin/bash
funWithParam(){
echo "The value of the first parameter is $1 !"
echo "The value of the second parameter is $2 !"
echo "The value of the tenth parameter is $10 !"
echo "The value of the tenth parameter is ${10} !"
echo "The value of the eleventh parameter is ${11} !"
echo "The amount of the parameters is $# !" # 参数个数
echo "The string of the parameters is $* !" # 传递给函数的全部参数
}
funWithParam 1 2 3 4 5 6 7 8 9 34 73
运行脚本:
The value of the first parameter is 1 !The value of the second parameter is 2 !The value of the tenth parameter is 10 !The value of the tenth parameter is 34 !The value of the eleventh parameter is 73 !The amount of the parameters is 12 !The string of the parameters is 1 2 3 4 5 6 7 8 9 34 73 !"
注意:$10 不能获取第十个参数,获取第十个参数须要${10}。当n>=10时,须要使用${n}来获取参数。
另外,还有几个特殊变量用来处理参数,前面已经提到:
特殊变量 |
说明 |
$# |
传递给函数的参数个数。 |
$* |
显示全部传递给函数的参数。 |
$@ |
与$*相同,可是略有区别,请查看Shell特殊变量。 |
$? |
函数的返回值。 |
二十3、Shell输入输出重定向:Shell Here Document,/dev/null文件
Unix 命令默认从标准输入设备(stdin)获取输入,将结果输出到标准输出设备(stdout)显示。通常状况下,标准输入设备就是键盘,标准输出设备就是终端,即显示器。
输出重定向
命令的输出不只能够是显示器,还能够很容易的转移向到文件,这被称为输出重定向。
命令输出重定向的语法为:
$ command > file
这样,输出到显示器的内容就能够被重定向到文件。
例如,下面的命令在显示器上不会看到任何输出:
$ who > users
打开 users 文件,能够看到下面的内容:
$ cat usersoko tty01 Sep 12 07:30ai tty15 Sep 12 13:32ruth tty21 Sep 12 10:10pat tty24 Sep 12 13:07steve tty25 Sep 12 13:03$
输出重定向会覆盖文件内容,请看下面的例子:
$ echo line 1 > users$ cat usersline 1ss$
若是不但愿文件内容被覆盖,可使用 >> 追加到文件末尾,例如:
$ echo line 2 >> users$ cat usersline 1line 2$
输入重定向
和输出重定向同样,Unix 命令也能够从文件获取输入,语法为:
command < file
这样,原本须要从键盘获取输入的命令会转移到文件读取内容。
注意:输出重定向是大于号(>),输入重定向是小于号(<)。
例如,计算 users 文件中的行数,可使用下面的命令:
$ wc -l users2 users$
也能够将输入重定向到 users 文件:
$ wc -l < users2$
注意:上面两个例子的结果不一样:第一个例子,会输出文件名;第二个不会,由于它仅仅知道从标准输入读取内容。
重定向深刻讲解
通常状况下,每一个 Unix/Linux 命令运行时都会打开三个文件:
标准输入文件(stdin):stdin的文件描述符为0,Unix程序默认从stdin读取数据。
标准输出文件(stdout):stdout 的文件描述符为1,Unix程序默认向stdout输出数据。
标准错误文件(stderr):stderr的文件描述符为2,Unix程序会向stderr流中写入错误信息。
默认状况下,command > file 将 stdout 重定向到 file,command < file 将stdin 重定向到 file。
若是但愿 stderr 重定向到 file,能够这样写:
$command 2 > file
若是但愿 stderr 追加到 file 文件末尾,能够这样写:
$command 2 >> file
2 表示标准错误文件(stderr)。
若是但愿将 stdout 和 stderr 合并后重定向到 file,能够这样写:
$command > file 2>&1
或
$command >> file 2>&1
若是但愿对 stdin 和 stdout 都重定向,能够这样写:
$command < file1 >file2
command 命令将 stdin 重定向到 file1,将 stdout 重定向到 file2。
所有可用的重定向命令列表 |
|
命令 |
说明 |
command > file |
将输出重定向到 file。 |
command < file |
将输入重定向到 file。 |
command >> file |
将输出以追加的方式重定向到 file。 |
n > file |
将文件描述符为 n 的文件重定向到 file。 |
n >> file |
将文件描述符为 n 的文件以追加的方式重定向到 file。 |
n >& m |
将输出文件 m 和 n 合并。 |
n <& m |
将输入文件 m 和 n 合并。 |
<< tag |
将开始标记 tag 和结束标记 tag 之间的内容做为输入。 |
Here Document
Here Document 目前没有统一的翻译,这里暂译为”嵌入文档“。Here Document 是 Shell 中的一种特殊的重定向方式,它的基本的形式以下:
command << delimiter
document
delimiter
它的做用是将两个 delimiter 之间的内容(document) 做为输入传递给 command。
注意:
结尾的delimiter 必定要顶格写,前面不能有任何字符,后面也不能有任何字符,包括空格和 tab 缩进。
开始的delimiter先后的空格会被忽略掉。
下面的例子,经过 wc -l 命令计算 document 的行数:
$wc -l << EOF This is a simple lookup program for good (and bad) restaurants in Cape Town.EOF3$
也能够 将 Here Document 用在脚本中,例如:
#!/bin/bash
cat << EOF
This is a simple lookup program
for good (and bad) restaurants
in Cape Town.
EOF
运行结果:
This is a simple lookup programfor good (and bad) restaurantsin Cape Town.
下面的脚本经过 vi 编辑器将 document 保存到 test.txt 文件:
#!/bin/sh
1.
filename=test.txt
vi $filename <<EndOfCommands
i
This file was created automatically from
a shell script
^[
ZZ
EndOfCommands
运行脚本:
$ sh test.shVim: Warning: Input is not from a terminal$
打开 test.txt,能够看到下面的内容:
$ cat test.txtThis file was created automatically froma shell script$
/dev/null 文件
若是但愿执行某个命令,但又不但愿在屏幕上显示输出结果,那么能够将输出重定向到 /dev/null:
$ command > /dev/null
/dev/null 是一个特殊的文件,写入到它的内容都会被丢弃;若是尝试从该文件读取内容,那么什么也读不到。可是 /dev/null 文件很是有用,将命令的输出重定向到它,会起到”禁止输出“的效果。
若是但愿屏蔽 stdout 和 stderr,能够这样写:
$ command > /dev/null 2>&1
像其余语言同样,Shell 也能够包含外部脚本,将外部脚本的内容合并到当前脚本。
Shell 中包含脚本可使用:
. filename
或
source filename
两种方式的效果相同,简单起见,通常使用点号(.),可是注意点号(.)和文件名中间有一空格。
例如,建立两个脚本,一个是被调用脚本 subscript.sh,内容以下:
url="http://see.xidian.edu.cn/cpp/view/2738.html"
一个是主文件 main.sh,内容以下:
#!/bin/bash
. ./subscript.sh
echo $url
执行脚本:
$chomd +x main.sh./main.shhttp://see.xidian.edu.cn/cpp/view/2738.html$
注意:被包含脚本不须要有执行权限。