1.建立和运行shell程序
什么是shell程序呢? 简单的说shell程序就是一个包含若干行 shell或者linux命令的文件. 像编写高级语言的程序一样,编写一个shell程序需要一个文本编辑器.如VI等. 在文本编辑环境下,依据shell的语法规则,输入一些shell/linux命令行,形成一个完整的程序文件. 执行shell程序文件有三种方法
(1)#chmod +x file
(2)#sh file
(3)# . file
在编写shell时,第一行一定要指明系统需要那种shell解释你的shell程序,如:#!/bin/bash Unix/Linux上常见的Shell脚本解释器有bash、sh、csh、ksh等,习惯上把它们称作一种Shell。我们常说有多少种Shell,其实说的是Shell脚本解释器。
- shell通配符
? 任意单个字符,不能匹配空
* 任意零个或多个字符组(不能匹配以点开头的文件)
[ab] a或者b
[a-z] a到z之间的任意字符,包括端点在内
是Shell 而非命令本身处理通配符,命令后的通配符会在命令执行前就被代换了
如果需要命令而非Shell处理通配符,请用“\"将通配符转义,跳脱字符
- bash中的引号
双引号 “ ” :允许通过$符号引用其他变量值,会把引号的内容当成整体来看待
单引号 ‘ ’ :禁止引用其他变量值,shell中特殊符号都被视为普通字符,会把引号的内容当成整体来看待
反撇号 `` : 会把命令执行的结果输出
$() 在执行命令的过程中会优先执行
; 连续执行命令
&& 可对一行命令进行分割,在执行过程中考虑上一个命令执行是否是正确的
|| 可对一行命令进行分割,在执行过程中不考虑上一个命令执行是否是正确的
! 命令历史
# echo "This system is "HOSTNAME""
This is system is HOSTNAME
# echo "This system is "$HISTNAME""
This is system is
# echo "This system is "$HOSTNAME""
This is system is desktop8.example.com
# echo 'This system is "$HOSTNAME"'
This is system is "$HOSTNAME"
[root@desktop8 ~]# echo $5.00
5.00
[root@desktop8 ~]# echo \$5.00
$5.00
echo `ls`cat aaa.sh ``
# 将cmd1执行结果作为cmd2参数,再将cmd2结果作为cmd3的参数
echo $(cat $(ls))
cmd3 $(cmd2 $(cmd1))
echo `cat ddd`ls``
ls && cat ddd
lsa || cat ddd
运算符
$(()) :运算符
$[]
# a=$((1 + 10))
# echo $a
# a=$[1 + 100]
# echo $a
[root@vagrant-centos65 data]# aaa=$((10-1))
[root@vagrant-centos65 data]# echo $aaa
9
[root@vagrant-centos65 data]# aaa=$((10*1))
[root@vagrant-centos65 data]# echo $aaa
10
[root@vagrant-centos65 data]# aaa=$((9/3))
[root@vagrant-centos65 data]# echo $aaa
3
[root@vagrant-centos65 data]# aaa=$((9%3))
[root@vagrant-centos65 data]# echo $aaa
0
[root@vagrant-centos65 data]# aaa=$((9**3))
[root@vagrant-centos65 data]# echo $aaa
729
- shell的配置文件(软件+配置文件)
用户加载shell配置流程:
user01-->login-->bash-->/etc/profile-->$HOME/.bash_profile-->$HOME/.bashrc-->/etc/bashrc
全局配置文件
2.shell中的变量
(1)常用系统变量 # :保存程序命令行参数的数目 ? :保存前一个命令的返回码 0 :保存程序名 * :以(“ 1 2…“)的形式保存所有输入的命令行参数 所有的参数会被当做一个字符串 @ :以(“ 1”” 2”…)的形式保存所有输入的命令行参数 所有的参数会以空格做分隔符单做一个字符串
(2)定义变量
shell语言是非类型的解释型语言,不象用C++/JAVA语言编程时需要事先声明变量.给一个变量赋值,实际上就是定义了变量.
在linux支持的所有shell中,都可以用赋值符号(=)为变量赋值.
如: abc=9
由于shell程序的变量是无类型的,所以用户可以使用同一个变量时而存放字符时而存放整数. 如:name=abc (bash/pdksh)
在变量赋值之后,只需在变量前面加一个 去引用.
echo $abc
删除变量:
unset abc
(3)位置变量
当运行一个支持多个命令行参数的shell程序时,这些变量的值将分别存放在位置变量里. 其中第一个参数存放在位置变量1,第二个参数存放在位置变量2,依次类推…,shell保留这些变量,不允许用户以令外的方式定义他们.同别的变量,用 符号引用他们. 总结:
- 变量的设置
- 变量的引用
- 删除变量
(4)内置bash中变量
$#:命令行中位置变量的个数
$*:所有位置变量的内容
$@: 所有位置参数的内容,与$*的分割任不一样,建议和所有的位置参数时使用这种方式。
$?:上一条命令执行后返回的状态,当返回状态值为0时表示执行正常,非0值表示执行异常或出错
$$:当前所在进程的进程号
$!:后台运行的最后一个进程号
$0:当前执行的进程/程序名
3.shell中引号的使用方法
shell使用引号(单引号/双引号)和反斜线(““)用于向shell解释器屏蔽一些特殊字符. 反引号(“)对shell则有特殊意义.
如: abc=”how are you” (bash/pdksh)
这个命令行把三个单词组成的字符串how are you作为一个整体赋值给变量abc.
abc1=’$LOGNAME,how are you!’ (bash/pdksh)
abc2=”$LOGNAME,how are you!” (bash/pdksh)
LOGNAME变量是保存当前用户名的shell变量,假设他的当前值是:wang.执行完两条命令后, abc1的内容是LOGNAME, how are you!.而abc2的内容是;wang, how are you!.
象单引号一样,反斜线也能屏蔽所有特殊字符.但是他一次只能屏蔽一个字符.而不能屏蔽 一组字符.
反引号的功能不同于以上的三种符号.他不具有屏蔽特殊字符的功能.但是可以通过他将一个命令的运行结果传递给另外一个命令.
如:
contents=ls
(bash/pdksh)
4.shell程序中的test命令
- 算术操作(expr) 在bash中只能做整数的运算
+ 加
- 减
* 乘
/ 除(取整)
% 取余
$(())
$[]
[root@vagrant-centos65 data]# aa=$((10-1))
[root@vagrant-centos65 data]# echo $aa
9
[root@vagrant-centos65 data]# bb=$[10/10]
[root@vagrant-centos65 data]# echo $bb
1
[root@vagrant-centos65 data]# cc=$[10+10]
[root@vagrant-centos65 data]# echo $cc
20
- 退出状态 在Linux系统中,每当命令执行完成后,系统都会返回一个退出状态。该退出状态用一整数值表示,用于判断命令运行正确与否。 若退出状态值为0,表示命令运行成功 若退出状态值不为0时,则表示命令运行失败 最后一次执行的命令的退出状态值被保存在内置变量“$?”中,所以可以通过echo语句进行测试命令是否运行成功 0 表示运行成功,程序执行未遇到任何问题 1~125 表示运行失败,脚本命令、系统命令错误或参数传递错误 126 找到了该命令但无法执行 >128 命令被系统强制结束
test命令 用途:测试特定的表达式是否成立,当条件成立时,命令执行后的返回值$?为0,否则为其他数值 格式: test 条件表达式 [ 条件表达式 ]
test可以测试表示有哪些: 1、文件状态 2、字符串的比对 3、整数的比对 4、多条件组合(|| && !)(-a -o !)
#man test
-a = && -o = || !
单个条件: [ ! 1 -eq 1 ] 两个值的结果再取反 ,感叹号的优先级别最低,除非加括号;
! [ 1 -eq 1 ]
多个条件:
[ 1 -eq 1 -a 10 -eq 10 ] = [ 1 -eq 1 ] && [ 10 -eq 10 ] 两个条件成立,才正确;
[ 1 -eq 1 -o 10 -eq 10 ] = [ 1 -eq 1 ] || [ 10 -eq 10 ] 取或;
[ 1 -eq 1 -o 10 -eq 10 -o 1 -eq 1 ] = [ 1 -eq 1 ] || [ 10 -eq 10 ] || [ 1 -eq 1 ]
-----------------------------------------------------
# test 1 = 1
# echo $?
0
# [ 1 = 1 ] --[] = test --单条件
# echo $?
0
# [ 1 = 1 ] && echo YES
YES
# [ 1 = 2 ] && echo YES --无返回
|| -o or
&& -a and
!
条件的组合:
[ 1 = 1 -a 1 = 2 ] 与组合
[ 1 = 1 ] && [ 1 = 2 ] 与组合
[ 1 = 1 ] || [ 1 = 2 ] 或组合
[ 1 = 1 -o 1 = 2 ] 或组合
在bash中,命令test用于计算一个条件表达式的值.他们经常在条件语句和循环语句中被用来判断某些条件是否满足. test命令的语法格式: test expression 或者 [expression]
- 逻辑运算符
逻辑运算符主要包括逻辑非、逻辑与、逻辑或运算符,具体描述如下表所示:
逻辑操作符 描述
!expression 如果expression为假,则测试结果为真
expression1 a expression2 如果expression1和expression同时为真,则测试结果为真
expression1 o expression2 如果expression1和expression2中有一个为真,则测试条件为真
在test命令中,可以使用很多shell的内部操作符.这些操作符介绍如下: (1)字符串操作符 用于计算字符串表达式 test命令 | 含义 Str1 = str2 | 当str1与str2相同时,返回True Str1! = str2| 当str1与str2不同时,返回True Str | 当str不是空字符时,返回True -n str | 当str的长度大于0时,返回True -z str | 当str的长度是0时,返回True
(2)整数操作符具有和字符操作符类似的功能.只是他们的操作是针对整数
test表达式 | 含义 |
Int1 -eq int2 | 当int1等于int2时,返回True |
Int1 -ge int2 | 当int1大于/等于int2时,返回True |
Int1 -le int2 | 当int1小于/等于int2时,返回True |
Int1 -gt int2 | 当int1大于int2时,返回True |
Int1 -ne int2 | 当int1不等于int2时,返回True |
(3)用于文件操作的操作符,他们能检查:文件是否存在,文件类型等
test表达式 | 含义 |
-d file | 当file是一个目录时,返回 True |
-f file | 当file是一个普通文件时,返回 True |
-r file | 当file是一个刻读文件时,返回 True |
-s file | 当file文件长度大于0时,返回 True |
-w file | 当file是一个可写文件时,返回 True |
-x file | 当file是一个可执行文件时,返回 True |
(4)shell的逻辑操作符用于修饰/连接包含整数,字符串,文件操作符的表达式
test表达式 | 含义 |
! expr | 当expr的值是False时,返回True |
Expr1 -a expr2 | 当expr1,expr2值同为True时,返回True |
Expr1 -o expr2 | 当expr1,expr2的值至少有一个为True时,返回True |
5.条件语句
同其他高级语言程序一样,复杂的shell程序中经常使用到分支和循环控制结构, bash,pdksh分别都有两种不同形式的条件语句:if语句和case语句.
(1)if语句
语法格式: bash/pdksh用法: if [expression1] then commands1 elif [expression2] commands2 else commands3 if 含义:当expression1的条件为True时,shell执行then后面的commands1命令;当 expression1的条件为false并且expression2的条件满足为True时,shell执行 commands2命令;当expression1和expressin2的条件值同为false时,shell执行 commands3命令.if语句以他的反写fi结尾.
(2)case语句
case语句要求shell将一个字符串S与一组字符串模式P1,P2,…,Pn比较,当S与 某个模式Pi想匹配时,就执行相应的那一部分程序/命令.shell的case语句中字符 模式里可以包含象*这样的通配符. 语法格式: bash/pdksh用法: case string1 in str1) commands1;; str2) commands2;; *) commands3;; esac 含义:shell将字符串string1分别和字符串模式str1和str2比较.如果string1与str1匹配,则shell执行commands1的命令/语句;如果string11和str2匹配,则shell执行commands2的命令/语句.否则shell将执行commands3的那段程序/命令.其中,每个分支的程序/命令都要以两个分号(;;)结束.
6.循环语句
当需要重复的某些操作时,就要用到循环语句
(1)for语句
大家知道在很多编程语言中for语句是最常见.在shell中也不例外.for语句要求shell将包含 在这个语句中的一组命令连续执行一定的次数. 语法格式: bash/pdksh 用法1: for var1 in list do commands done 含义:在这个for语句中,对应于list中的每个值,shell将执行一次commands代表的一组命令. 在整个循环的每一次执行中,变量var1将依此取list中的不同的值. 用法2: for var1 do setatements done 含义:在这个for语句中,shell针对变量var1中的每一项分别执行一次statements代表的一组 命令.当使用这种形式的语句时,shell认为var1变量中包含了所有的位置变量,而位置变量中 存放着程序的命令行参数值.也就是说,他等价于下列形式: for var1 in “ @” do statements done 举例: for file ;bash/pdksh do tr a-z A-Z< file>file.caps done
(2)while语句
while语句是shell提供的另一种循环语句. while语句指定一个表达式和一组命令.这个语句使得shell重复执行一组命令,直到表达式的值为False为止. 语法格式: while expression ;bash do statements done
举例:
#!/bin/bash
count=1
while true
do
echo "this is a parameter number $count $1"
shift
count=$[ $count+1 ]
done
语句中shift命令的功能是将所有的命令行参数依次相左传递.
7.shell中的函数
1、在编写Shell脚本程序时,将一些需要重复使用的命令操作,定义为公共使用的语句块,即可称为函数 2、合理使用Shell函数,可以使脚本内容更加简洁,增强程序的易读性,提高执行效率 shell允许用户定义自己的函数.函数是高级语言中的重要结构.shell中的函数于C或 者其他 语言中定义的函数一样.与从头开始,一行一行地写程序相比,使用函数主要好处是有 利于组织 整个程序.在bash中,一个函数的语法格式如下: fname (){ shell comands } 定义好函数后,需要在程序中调用他们.bash中调用函数的格式: fname [parm1 parm2 parm3…] 调用函数时,可以向函数传递任意多个参数.函数将这些参数看做是存放他的命令行参数的 位置变量 总结 利用shell编程是提高系统管理工作效率的重要手段,学好shell跟了解系统基本命令 和管理工具的使用方法同样重要! 附:
A.bash中常用的命令
命令 | 含义 & 把程序放到后台执行 ctrl + z 可以将一个正在前台执行的命令放到后台,并且暂停 Alias |设置命令别名 Bg |将一个被挂起的进程在后台执行 cd |改变用户的当前目录 exit |终止一个shell export |使作为这个命令的参数的变量及其当前值,在当前运行的shell的子进程中可见 fc |编辑当前的命令行历史列表 fg |让一个被挂起的进程在前台执行 help |显示bash内部命令的帮助信息 history |显示最近输入的一定数量的命令行 kill |终止一个进程 pwd |显示用户当前工作目录 unalias |删除命令行别名
B.bash中常用的系统变量
变量 | 含义 |
EDITOR,FCEDIT | Bash的fc命令的默认文本编辑器 |
HISTFILE | 规定存放最近输入命令行文件的名字 |
HISTSIZE | 规定命令行历史文件的大小 |
HOME | 当前用户的宿主目录 |
OLDPWD | 用户使用的前一个目录 |
PATH | 规定bash寻找可执行文件时搜索的路径 |
PS1 | 命令行环境中显示第一级提示符号 |
PS2 | 命令行环境中显示第二级提示符号 |
PWD | 用户当前工作目录 |
SECONDS | 当前运行的bash进程的运行时间(以秒为单位)一 |
排序统计相关的:wc sort uniq
wc
-l 显示行数
-w 显示单词数
-m 显示字符数
默认不加参数,就是相当于上面三个参数都加
[root@li shell01]# cat /etc/passwd | wc -l |
79
[root@li shell01]# cat /etc/passwd | wc -w |
106
[root@li shell01]# cat /etc/passwd | wc -m |
3374
[root@li shell01]# cat /etc/passwd | wc |
79 106 3374
wc -L 算一个文件里最长一行有多少个字符
sort 排序命令
[root@li shell01]# cat /etc/passwd |
sort –默认以开头字母排序
-r 反向排序
-n 以数字来排
-f 大小写不敏感
-t 分隔符
-k 接数字代表第几列
cut [-bn] [file] 或 cut [-c] [file] 或 cut [-df] [file]
cut 命令从文件的每一行剪切字节、字符和字段并将这些字节、字符和字段写至标准输出。
如果不指定 File 参数,cut 命令将读取标准输入。必须指定 -b、-c 或 -f 标志之一。
-b :以字节为单位进行分割。这些字节位置将忽略多字节字符边界,除非也指定了 -n 标志。
-c :以字符为单位进行分割。
-d :自定义分隔符,默认为制表符。
-f :与-d一起使用,指定显示哪个区域。
-n :取消分割多字节字符。仅和 -b 标志一起使用。如果字符的最后一个字节落在由 -b 标志的 List 参数指示的
范围之内,该字符将被写出;否则,该字符将被排除。
[root@li ~]# cut -d -f
[root@li ~]# sort -t -k
[root@li ~]# awk -F n
[root@li shell01]# cat /etc/passwd | sort -t “:” -k 3 –以UID来排序,但是它只会以数字的第一个数字来排也就是说 2要排到14的后面 |
[root@li shell01]# cat /etc/passwd | sort -t “:” -k 3 -n –多加一个-n参数,才会以整个的数字大小来排序 |
uniq 唯一命令
默认是以连续的重复值内只取一个
[root@li shell01]# cat /etc/passwd | cut -d “:” -f7 | uniq | grep bash |
/bin/bash
/bin/bash
/bin/bash
/bin/bash
[root@li shell01]# cat /etc/passwd | cut -d “:” -f7 | grep bash | uniq |
/bin/bash
–在管道用得多的情况下,命令的顺序会造成很大的结果不同
---------------------------------
练习:
对有下面内容的文件进行操作
http://a.domain.com/1.html
http://b.domain.com/1.html
http://c.domain.com/1.html
http://a.domain.com/2.html
http://a.domain.com/3.html
http://b.domain.com/3.html
http://c.domain.com/2.html
http://c.domain.com/3.html
http://a.domain.com/1.html
得到下面的结果
4 a.domain.com
3 c.domain.com
2 b.domain.com
cat 1.txt | cut -d”/” -f3 | sort | uniq -c | sort -n -r |
- bash的调试方法
打开调试模式
• $set -x
• $bash -x shell.sh
• 关闭调试模式
• $set +x
sh -n test.sh --检查语法(结构化语句的语法)
sh -x test.sh --调试
如果参数过太多,出现这个报错:too many arguments,需要使用xargs
xargs (1) - build and execute command lines from standard input
# awk -F: '{print $1}' /etc/passwd | xargs -i mkdir -p {}