Shell学习笔记.docx
《Shell学习笔记.docx》由会员分享,可在线阅读,更多相关《Shell学习笔记.docx(29页珍藏版)》请在冰豆网上搜索。
Shell学习笔记
好后悔在学校里,没有充分利用时间。
一些基础知识、基础技能都没有涉及到。
现在用到了,才感慨万千。
希望从现在开始study,还来得及。
好的,既然觉悟了,那就从Shell基础开始吧
。
1、Shell简介
①、Shell既是一种命令语言,又是一种程序设计语言。
作为命令语言,它交互式地解释和执行用户输入的命令;
作为程序设计语言,它定义了各种变量和参数,并提供了许多在高级语言中才具有的控制结构,包括循环和分支。
②、Shell有两种执行命令的方式:
交互式(Interactive):
解释执行用户的命令,用户输入一条命令,Shell就解释执行一条。
批处理(Batch):
用户事先写一个Shell脚本(Script),其中有很多条命令,让Shell一次把这些命令执行完,而不必一条一条地敲命令。
2、Shell的种类
Shell脚本解释器有bash、sh、csh、ksh等。
bash是Linux标准默认的shell。
bash由BrianFox和ChetRamey共同完成,是BourneAgainShell的缩写,内部命令一共有40个。
3、编译型语言和解释性语言
编译型语言:
C、C++、java等。
预先将我们写好的源代码(sourcecode)转换成目标代码(objectcode),这个过程被称作“编译”。
目标代码接近计算机底层,故执行效率高。
解释性语言:
awk、Perl、Python、Ruby与Shell,也被称为脚本语言。
执行这类程序时,解释器(interpreter)需要读取我们编写的源代码(sourcecode),并将其转换成目标代码(objectcode),再由计算机运行。
因为每次执行程序都多了编译的过程,因此效率有所下降。
4、Shell的优点:
简单性:
Shell是一个高级语言;通过它,你可以简洁地表达复杂的操作。
可移植性:
使用POSIX所定义的功能,可以做到脚本无须修改就可在不同的系统上执行。
开发容易:
可以在短时间内完成一个功能强大又妤用的脚本。
5、Shell注释
新建一个文件,扩展名为sh(sh代表shell),扩展名并不影响脚本执行,见名知意就好,如果你用php写shell脚本,扩展名就用php。
Shell注释:
以“#”开头的行就是注释,会被解释器忽略。
sh里没有多行注释,只能每一行加一个#号。
如果在开发过程中,遇到大段的代码需要临时注释起来,过一会儿又取消注释,怎么办呢?
每一行加个#符号太费力了,可以把这一段要注释的代码用一对花括号括起来,定义成一个函数,没有地方调用这个函数,这块代码就不会执行,达到了和注释一样的效果。
6、Shell的运行方式
#!
/bin/bash---“#!
”是一个约定的标记,它告诉系统这个脚本需要什么解释器来执行,即使用哪一种Shell。
echo"HelloWorld!
"---echo命令用于向窗口输出文本。
运行脚本有两种方式:
1作为可执行程序:
chmod+x./test.sh#使脚本具有执行权限,“./”表示在当前目录找。
./test.sh#执行脚本
注意,一定要写成./test.sh,而不是test.sh。
运行其它二进制的程序也一样,直接写test.sh,linux系统会去PATH里寻找有没有叫test.sh的,而只有/bin,/sbin,/usr/bin,/usr/sbin等在PATH里,你的当前目录通常不在PATH里,所以写成test.sh是会找不到命令的,要用./test.sh告诉系统说,就在当前目录找。
2作为解释器参数:
这种运行方式是,直接运行解释器,其参数就是shell脚本的文件名,如:
/bin/shtest.sh
这种方式运行的脚本,不需要在第一行指定解释器信息,写了也与前面的没关系。
7、Shell变量
自定义变量:
1、your_name="mozhiyan"----变量名和等号之间不能有空格
2、echo$your_name-----“$”表示使用变量
3、echo${your_name}----推荐使用
--2、3是一个意思。
变量名外面的花括号是可选的,加不加都行,加花括号是为了帮助解释器识别变量的边界
只读变量:
将变量定义为只读变量,只读变量的值不能被改变
readonlyyour_name
删除变量:
删除的是变量的值。
变量将没有任何输出。
unsetvariable_name
变量类型
运行shell时,会同时存在三种变量:
1)局部变量:
局部变量在脚本或命令中定义,仅在当前shell实例中有效,其他shell启动的程序不能访问局部变量。
2)环境变量:
所有的程序,包括shell启动的程序,都能访问环境变量,有些程序需要环境变量来保证其正常运行。
必要的时候shell脚本也可以定义环境变量。
3)shell变量:
shell变量是由shell程序设置的特殊变量。
shell变量中有一部分是环境变量,有一部分是局部变量,这些变量保证了shell的正常运行
特殊变量:
$表示当前Shell进程的ID,即pid,看下面的代码:
$echo$$
运行结果:
29949
特殊变量列表
变量
含义
$0
当前脚本的文件名
$n
传递给脚本或函数的参数。
n是一个数字,表示第几个参数。
例如,第一个参数是$1,第二个参数是$2。
$#
传递给脚本或函数的参数个数。
$*
传递给脚本或函数的所有参数。
$@
传递给脚本或函数的所有参数。
被双引号("")包含时,与$*稍有不同,下面将会讲到。
$?
上个命令的退出状态,或函数的返回值。
退出状态是一个数字,一般情况下,大部分命令执行成功会返回0,失败返回1,也有一些命令返回其他值,表示不同类型的错误
$$
当前Shell进程ID。
对于Shell脚本,就是这些脚本所在的进程ID。
命令行参数:
运行脚本时传递给脚本的参数称为命令行参数。
命令行参数用$n表示,例如,$1表示第一个参数,$2表示第二个参数,依次类推。
$*和$@的区别
$*和$@都表示传递给函数或脚本的所有参数,不被双引号("")包含时,都以"$1""$2"…"$n"的形式输出所有参数。
但是当它们被双引号("")包含时,"$*"会将所有的参数作为一个整体,以"$1$2…$n"的形式输出所有参数;"$@"会将各个参数分开,以"$1""$2"…"$n"的形式输出所有参数。
8、Shell替换
如果表达式中包含特殊字符,Shell将会进行替换。
例如,在双引号中使用变量就是一种替换,转义字符也是一种替换。
#!
/bin/bash
a=10
echo-e"Valueofais$a\n"
运行结果:
Valueofais10
这里-e表示对转义字符进行替换。
如果不使用-e选项,将会原样输出:
Valueofais10\n
echo命令的-E选项禁止转义,默认也是不转义的;使用-n选项可以禁止插入换行符。
命令替换:
命令替换是指Shell可以先执行命令,将输出结果暂时保存,在适当的地方输出。
变量替换:
变量替换可以根据变量的状态(是否为空、是否定义等)来改变它的值
可以使用的变量替换形式:
形式
说明
${var}
变量本来的值
${var:
-word}
如果变量var为空或已被删除(unset),那么返回word,但不改变var的值。
${var:
=word}
如果变量var为空或已被删除(unset),那么返回word,并将var的值设置为word。
${var:
?
message}
如果变量var为空或已被删除(unset),那么将消息message送到标准错误输出,可以用来检测变量var是否可以被正常赋值。
若此替换出现在Shell脚本中,那么脚本将停止运行。
${var:
+word}
如果变量var被定义,那么返回word,但不改变var的值。
9、Shell运算符
Bash支持很多运算符,包括算数运算符、关系运算符、布尔运算符、字符串运算符和文件测试运算符。
原生bash不支持简单的数学运算,但是可以通过其他命令来实现,例如awk和expr,expr最常用。
expr是一款表达式计算工具,使用它能完成表达式的求值操作。
#!
/bin/bash
val=`expr2+2`
echo"Totalvalue:
$val"
两点注意:
1、表达式和运算符之间要有空格,例如2+2是不对的,必须写成2+2,这与我们熟悉的大多数编程语言不一样。
2、完整的表达式要被``包含,注意这个字符不是常用的单引号,在Esc键下边。
算术运算符列表
运算符
说明
举例
+
加法
`expr$a+$b`结果为30。
-
减法
`expr$a-$b`结果为10。
*
乘法
`expr$a\*$b`结果为200。
/
除法
`expr$b/$a`结果为2。
%
取余
`expr$b%$a`结果为0。
=
赋值
a=$b将把变量b的值赋给a。
==
相等。
用于比较两个数字,相同则返回true。
[$a==$b]返回false。
!
=
不相等。
用于比较两个数字,不相同则返回true。
[$a!
=$b]返回true。
乘号(*)前边必须加反斜杠(\)才能实现乘法运算;
注意:
条件表达式要放在方括号之间,并且要有空格,例如[$a==$b]是错误的,必须写成[$a==$b]。
关系运算符:
关系运算符只支持数字,不支持字符串,除非字符串的值是数字。
关系运算符列表
运算符
说明
举例
-eq
检测两个数是否相等,相等返回true。
[$a-eq$b]返回true。
-ne
检测两个数是否相等,不相等返回true。
[$a-ne$b]返回true。
-gt
检测左边的数是否大于右边的,如果是,则返回true。
[$a-gt$b]返回false。
-lt
检测左边的数是否小于右边的,如果是,则返回true。
[$a-lt$b]返回true。
-ge
检测左边的数是否大等于右边的,如果是,则返回true。
[$a-ge$b]返回false。
-le
检测左边的数是否小于等于右边的,如果是,则返回true。
[$a-le$b]返回true。
布尔运算符:
布尔运算符列表
运算符
说明
举例
!
非运算,表达式为true则返回false,否则返回true。
[!
false]返回true。
-o
或运算,有一个表达式为true则返回true。
[$a-lt20-o$b-gt100]返回true。
-a
与运算,两个表达式都为true才返回true。
[$a-lt20-a$b-gt100]返回false。
字符串运算符列表:
字符串运算符列表
运算符
说明
举例
=
检测两个字符串是否相等,相等返回true。
[$a=$b]返回false。
!
=
检测两个字符串是否相等,不相等返回true。
[$a!
=$b]返回true。
-z
检测字符串长度是否为0,为0返回true。
[-z$a]返回false。
-n
检测字符串长度是否为0,不为0返回true。
[-z$a]返回true。
str
检测字符串是否为空,不为空返回true。
[$a]返回true。
文件测试运算符:
文件测试运算符用于检测Unix文件的各种属性。
Fileexists
文件测试运算符列表
操作符
说明
举例
-bfile
检测文件是否是块设备文件,如果是,则返回true。
[-b$file]返回false。
-cfile
检测文件是否是字符设备文件,如果是,则返回true。
[-b$file]返回false。
-dfile
检测文件是否是目录,如果是,则返回true。
[-d$file]返回false。
-ffile
检测文件是否是普通文件(既不是目录,也不是设备文件),如果是,则返回true。
[-f$file]返回true。
-gfile
检测文件是否设置了SGID位,如果是,则返回true。
[-g$file]返回false。
-kfile
检测文件是否设置了粘着位(StickyBit),如果是,则返回true。
[-k$file]返回false。
-pfile
检测文件是否是具名管道,如果是,则返回true。
[-p$file]返回false。
-ufile
检测文件是否设置了SUID位,如果是,则返回true。
[-u$file]返回false。
-rfile
检测文件是否可读,如果是,则返回true。
[-r$file]返回true。
-wfile
检测文件是否可写,如果是,则返回true。
[-w$file]返回true。
-xfile
检测文件是否可执行,如果是,则返回true。
[-x$file]返回true。
-sfile
检测文件是否为空(文件大小是否大于0),不为空返回true。
[-s$file]返回true。
-efile
检测文件(包括目录)是否存在,如果是,则返回true。
[-e$file]返回true。
10、Shell字符串
字符串可以用单引号,也可以用双引号,也可以不用引号。
单引号:
str='thisisastring'
单引号字符串的限制:
1)单引号里的任何字符都会原样输出,单引号字符串中的变量是无效的;
2)单引号字串中不能出现单引号(对单引号使用转义符后也不行)。
双引号:
your_name='qinjx'
str="Hello,Iknowyourare\"$your_name\"!
\n"
双引号的优点:
1、双引号里可以有变量
2、双引号里可以出现转义字符
拼接字符串:
your_name="qinjx"
greeting="hello,"$your_name"!
"
greeting_1="hello,${your_name}!
"
echo$greeting$greeting_1
获取字符串长度:
string="abcd"
echo${#string}#输出4
提取子字符串:
string="alibabaisagreatcompany"
echo${string:
1:
4}#输出liba
查找子字符串:
string="alibabaisagreatcompany"
echo`exprindex"$string"is`
定义数组:
在Shell中,用括号来表示数组,数组元素用“空格”符号分割开。
定义数组的一般形式为:
array_name=(value1...valuen)
array_name=(value0value1value2value3)
读取数组元素值的一般格式是:
${array_name[index]}
例如:
valuen=${array_name[2]}
使用@或*可以获取数组中的所有元素,例如:
${array_name[*]}
${array_name[@]}
获取数组的长度:
获取数组长度的方法与获取字符串长度的方法相同,例如:
#取得数组元素的个数
length=${#array_name[@]}
#或者
length=${#array_name[*]}
#取得数组单个元素的长度
lengthn=${#array_name[n]}
11、echo命令
显示转义字符:
echo"\"Itisatest\""
结果将是:
"Itisatest"
显示结果重定向至文件:
echo"Itisatest">myfile
原样输出字符串:
若需要原样输出字符串(不进行转义),请使用单引号。
例如:
echo'$name\"'
显示命令执行结果:
echo`date`
从上面可看出,双引号可有可无,单引号主要用在原样输出中。
12、printf命令
printf命令用于格式化输出,是echo命令的增强版。
它是C语言printf()库函数的一个有限的变形,并且在语法上有些不同。
printf命令的语法:
printfformat-string[arguments...]
format-string为格式控制字符串,arguments为参数列表。
这里仅说明与C语言printf()函数的不同:
1、printf命令不用加括号
2、format-string可以没有引号,但最好加上,单引号双引号均可。
3、参数多于格式控制符(%)时,format-string可以重用,可以将所有参数都转换。
4、Arguments使用空格分隔,不用逗号。
请看下面的例子:
#format-string为双引号
$printf"%d%s\n"1"abc"
1abc
#单引号与双引号效果一样
$printf'%d%s\n'1"abc"
1abc
#没有引号也可以输出
$printf%sabcdef
abcdef
#格式只指定了一个参数,但多出的参数仍然会按照该格式输出,format-string被重用
$printf%sabcdef
abcdef
$printf"%s\n"abcdef
abc
def
$printf"%s%s%s\n"abcdefghij
abc
def
ghi
j
#如果没有arguments,那么%s用NULL代替,%d用0代替
$printf"%sand%d\n"
and0
#如果以%d的格式来显示字符串,那么会有警告,提示无效的数字,此时默认置为0
$printf"Thefirstprogramalwaysprints'%s,%d\n'"HelloShell
-bash:
printf:
Shell:
invalidnumber
Thefirstprogramalwaysprints'Hello,0'
13、Shellifelse语句
if语句通过关系运算符判断表达式的真假来决定执行哪个分支。
Shell有三种if...else语句:
1、if...fi语句;
2、if...else...fi语句;
3、if...elif...else...fi语句。
1)if...else语句
if...else语句的语法:
if[expression]
then
Statement(s)tobeexecutedifexpressionistrue
fi
如果expression返回true,then后边的语句将会被执行;如果返回false,不会执行任何语句。
最后必须以fi来结尾闭合if,fi就是if倒过来拼写,后面也会遇见。
注意:
expression和方括号([])之间必须有空格,否则会有语法错误。
2)if...else...fi语句
if...else...fi语句的语法:
if[expression]
then
Statement(s)tobeexecutedifexpressionistrue
else
Statement(s)tobeexecutedifexpressionisnottrue
fi
如果expression返回true,那么then后边的语句将会被执行;否则,执行else后边的语句。
3)if...elif...fi语句
if...elif...fi语句可以对多个条件进行判断,语法为:
if[expression1]
then
Statement(s)tobeexecutedifexpression1istrue
elif[expression2]
then
Statement(s)tobeexecutedifexpression2istrue
elif[expression3]
then
Statement(s)tobeexecutedifexpression3istrue
else
Statement(s)tobeexecutedifnoexpressionistrue
fi
哪一个expression的值为true,就执行哪个expression后面的语句;如果都为false,那么不执行任何语句。
if...else语句也可以写成一行,以命令的方式来运行,像这样:
iftest$[2*3]-eq$[1+5];thenecho'Thetwonumbersareequal!
';fi;
if...else语句也经常与test命令结合使用,如下所示:
num1=$[2*3]
num2=$[1+5]
iftest$[num1]-eq$[num2]
then
echo'Thetwonumbersareequal!
'
else
echo'Thetwonumbersarenotequal!
'
fi
test命令用于检查某个条件是否成立,与方括号([])类似。
14、Shellcaseesac语句
case...esac与其他语言中的switch...case语句类似,是一种多分枝选择结构。
case语句匹配一个值或一个模式,如果匹配成功,执行相匹配的命令。
case语句格式如下:
case值in
模式1)
command1
command2
command3
;;
模式2)
command1
command2
command3
;;
*)
command1
command2
command3
;;
esac
case工作方式如上所示。
取值后面必须为关键字in,每一模式必须以右括号结束。
取值可以为变量或常数。
匹配发现取值符合某一模式后,其间所有命令开始执行直至;;。
;;与其他语言中的