Shell 环境
Shell 编程跟 JavaScript、php 编程一样,只要有一个能编写代码的文本编辑器和一个能解释执行的脚本解释器就可以了。Linux 的 Shell 种类众多,常见的有:
• Bourne Shell(/usr/bin/sh或/bin/sh)
• Bourne Again Shell(/bin/bash) 本教程关注于此
• C Shell(/usr/bin/csh)
• K Shell(/usr/bin/ksh)
• Shell for Root(/sbin/sh)
• ……
第一个shell脚本
新建一个文件 test.sh
#!/bin/bash #! 是一个约定的标记,它告诉系统这个脚本需要什么解释器来执行,即使用哪一种 Shell
echo "Hello World !" echo 命令用于向窗口输出文本。
运行:
1、作为可执行程序: chmod +x ./test.sh #使脚本具有执行权限 ./test.sh #执行脚本
2、作为解释器参数:/bin/sh test.sh
Shell 变量
your_name="runoob.com" 定义变量,不能有空格
echo "hello $your_name" 使用变量
echo ${your_name} 使用变量
readonly your_name 把变量变为只读的
unset your_name 删除变量,不能删除只读变量
for file in `ls /etc` 除了显式地直接赋值,还可以用语句给变量赋值
for file in $(ls /etc) 除了显式地直接赋值,还可以用语句给变量赋值
Shell 字符串
your_name="runoob"
greeting="hello, "$your_name" !" 字符串拼接
greeting_1="hello, ${your_name} !" 双引号不拼接也能显示变量
greeting_2='hello, '$your_name' !'
greeting_3='hello, ${your_name} !' 单引号不拼接不能显示变量单引号字符串中的变量是无效的
string="runoob is a great site"
echo ${#string} #输出 22 提取字符串长度
echo ${string:1:4} # 输出 unoo 提取子字符串
echo `expr index "$string" io` # 输出 4 查找字符 i 或 o 的位置(哪个字母先出现就计算哪个):
Shell 数组
数组名=(值1 值2 ... 值n) 定义数组
array_name=(value0 value1 value2 value3)
array_name[0]=value0 单独定义数组的各个分量:
array_name[1]=value1
array_name[n]=valuen
echo ${数组名[下标]} 读取数组元素值
echo ${array_name[@]} 获取数组中的所有元素
length=${#array_name[@]} 取得数组元素的个数
length=${#array_name[*]} 取得数组元素的个数
lengthn=${#array_name[n]} 取得数组单个元素的长度
Shell多行注释
多行注释还可以使用以下格式 EOF可以用其他符号:
:<<EOF
注释内容...
注释内容...
注释内容...
EOF
Shell 传递参数
echo "执行的文件名:$0";
echo "第一个参数为:$1";
echo "第二个参数为:$2";
echo "参数个数为:$#";
echo "传递的参数作为一个字符串显示:$*";
echo "传递的参数作为多个字符串显示:$@";
$$ 脚本运行的当前进程ID号
$! 后台运行的最后一个进程的ID号
$- 显示Shell使用的当前选项,与set命令功能相同。
$? 显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误。
for i in "$*"; do
echo $i
done
for i in "$@"; do
echo $i
done
执行脚本,输出结果如下所示:
$ chmod +x test.sh
$ ./test.sh 1 2 3
-- $* 演示 ---
1 2 3
-- $@ 演示 ---
1
2
3
Shell 基本运算符
- 算数运算符、关系运算符、布尔运算符、字符串运算符、文件测试运算符
- 原生bash不支持简单的数学运算,但是可以通过其他命令来实现,例如 awk 和 expr,expr 最常用。expr 是一款表达式计算工具,使用它能完成表达式的求值操作。
算术运算符
a=10 这也是算数运算符,注意没有空格,其他算数运算符都有空格 b=20 val=`expr $a + $b`
val=`expr $a - $b` val=`expr $a \* $b` 注意要转义一下 val=`expr $b / $a` val=`expr $b % $a` if [ $a == $b ] 条件表达式要放在方括号之间,并且要有空格 then echo "a 等于 b" fi if [ $a != $b ] then echo "a 不等于 b" fi
关系运算符
if [ $a -eq $b ] 等于
if [ $a -ne $b ] 不等于
if [ $a -gt $b ] 大于
if [ $a -lt $b ] 小于
if [ $a -ge $b ] 大于或等于
if [ $a -le $b ] 小于或等于
布尔运算符
a=10 b=20 if [ $a != $b ] 不等于 if [ $a -lt 100 -a $b -gt 15 ] 逻辑与 if [ $a -lt 100 -o $b -gt 100 ] 逻辑或
逻辑运算符
if [[ $a -lt 100 && $b -gt 100 ]] 注意是[[]]
if [[ $a -lt 100 || $b -gt 100 ]]
字符串运算符
a="abc" b="efg" if [ $a = $b ] 检测两个字符串是否相等 if [ $a != $b ] 检测两个字符串是否不相等
if [ $a ] 检测字符串是否不为空if [ -z $a ] 检测字符串长度是否为0
if [ -n $a ] 检测字符串长度是否不为 0
文件测试运算符
file="/var/www/runoob/test.sh"
if [ -b $file ] 检测文件是否是块设备文件,如果是,则返回 true。
if [ -c $file ] 检测文件是否是字符设备文件,如果是,则返回 true。
if [ -d $file ] 检测文件是否是目录,如果是,则返回 true。
if [ -f $file ] 检测文件是否是普通文件(既不是目录,也不是设备文件),如果是,则返回 true。
if [ -g $file ] 检测文件是否设置了 SGID 位,如果是,则返回 true。
if [ -k $file ] 检测文件是否设置了粘着位(Sticky Bit),如果是,则返回 true。
if [ -p $file ] 检测文件是否是有名管道,如果是,则返回 true。
if [ -u $file ] 检测文件是否设置了 SUID 位,如果是,则返回 true。
if [ -r $file ] 检测文件是否可读,如果是,则返回 true。
if [ -w $file ] 检测文件是否可写,如果是,则返回 true。
if [ -x $file ] 检测文件是否可执行,如果是,则返回 true。
if [ -s $file ] 检测文件是否为空(文件大小是否大于0),不为空返回 true。
if [ -e $file ] 检测文件(包括目录)是否存在,如果是,则返回 true。
echo命令
echo "\"It is a test\"" 显示转义字符
echo -e "OK! \n It is a test " \n 换行
echo -e "OK! \c" -e 开启转义 \c 不换行
echo "It is a test"
echo "It is a test" > myfile 显示结果定向至文件
echo '$name\"'原样输出字符串,不进行转义或取变量(用单引号)
echo `date` 显示命令执行结果
printf 命令比使用 echo 移植性好
$ echo "Hello, Shell" $ printf "Hello, Shell\n" printf不自动加换行
printf "%-10s %-8s %-4.2f\n" 郭靖 男 66.1234
printf "%d %s\n" 1 "abc"
printf "%s\n" abc def 格式只指定了一个参数,但多出的参数仍然会按照该格式输出,format-string 被重用
printf "%s %s %s\n" a b c d e f g h i j
printf %s abcdef 没有引号也可以输出
%s 输出一个字符串,%d 整型输出,%c 输出一个字符,%f 输出实数
%-10s 指一个宽度为 10 个字符(- 表示左对齐,没有则表示右对齐)
%-4.2f 指格式化为小数,其中 .2 指保留2位小数
\a 警告字符,通常为ASCII的BEL字符
\b 后退
\c 抑制(不显示)输出结果中任何结尾的换行字符(只在%b格式指示符控制下的参数字符串中有效),而且,任何留在参数里的字符、任何接下来的参数以及任何留在格式字符串中的字符,都被忽略
\f 换页(formfeed)
\n 换行
\r 回车(Carriage return)
\t 水平制表符
\v 垂直制表符
\\ 一个字面上的反斜杠字符
\ddd 表示1到3位数八进制值的字符。仅在格式字符串中有效
\0ddd 表示1到3位的八进制值字符
test 命令用于检查某个条件是否成立,它可以进行数值、字符和文件三个方面的测试
num1=100 num2=100 if test $[num1] -eq $[num2] -eq –ne –gt –ge –lt –le数值测试, 注意是中括号 result=$[num1+num2] 代码中的 [] 执行基本的算数运算
num1="ru1noob" num2="runoob" if test $num1 = $num2 != -z –n 字符串测试
if test $num1 != $num2
if test –z $num1
if test –n $num1
if test -e ./bash -e –r –w –x –s –d –f –c –b 文件测试
if test -e ./notFile -o -e ./bash -o –a !逻辑运算
流程控制流程控制不可为空
if condition1
then
command1
elif condition2
then
command2
else
commandN
fi
if [ $(ps -ef | grep -c "ssh") -gt 1 ]; then echo "true"; fi 写成一行
for var in item1 item2 ... itemN
do
command1
...
done
for var in item1 item2 ... itemN; do command1; command2… done; 写成一行
for loop in 1 2 3 4 5
do
echo "The value is: $loop"
done
for str in This is a string
do
echo $str
done
while condition
do
command
done
int=1
while(( $int<=5 ))
do
echo $int
let "int++"
done
echo '按下 <CTRL-D> 退出' echo -n '输入你最喜欢的网站名: ' while read FILM do echo "是的!$FILM 是一个好网站" done
while : 无限循环
do
command
done
while true 无限循环
do
command
done
until condition
do
command
done
a=0 until [ ! $a -lt 10 ] do echo $a a=`expr $a + 1` done
case 值 in
模式1)
command1
command2
...
commandN
;;
模式2)
command1
command2
...
commandN
;;
Esac
read aNum case $aNum in 1) echo '你选择了 1' ;; 2) echo '你选择了 2' ;; 3) echo '你选择了 3' ;; 4) echo '你选择了 4' ;; *) echo '你没有输入 1 到 4 之间的数字' ;; esac
site="runoob" case "$site" in "runoob") echo "菜鸟教程" ;; "google") echo "Google 搜索" ;; "taobao") echo "淘宝网" ;; esac
break 、 continue 用于循环
函数
demoFun(){
echo "这是我的第一个 shell 函数!"
}
echo "-----函数开始执行-----"
demoFun
echo "-----函数执行完毕-----"
funWithReturn(){ 函数要先定义再使用
echo "输入第一个数字: "
read aNum
echo "输入第二个数字: "
read anotherNum
return $(($aNum+$anotherNum))
}
funWithReturn
echo "输入的两个数字之和为 $?" 函数返回值在调用该函数后通过 $? 来获得。
funWithParam(){
echo "第一个参数为 $1 !"
echo "第二个参数为 $2 !"
echo "第十个参数为 $10 !" 注意大于1位数获取不到
echo "第十个参数为 ${10} !"
echo "第十一个参数为 ${11} !"
echo "参数总数有 $# 个!"
echo "作为一个字符串输出所有参数 $* !"
}
funWithParam 1 2 3 4 5 6 7 8 9 34 73
输入/输出重定向
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 合并。
command << tag
Document 将tag 之间的内容作为输入。
Tag
command > /dev/null 执行某个命令,但又不希望在屏幕上显示输出结果
$cat a.txt > b 会把a.txt中的内容输出到b中
$cat a.txt >> b b中就会有2个a.txt中的内容了
$echo “nihao” > b
$wc –l a.txt 统计文件的行数,输出 12 a.txt
$wc –l < a.txt 统计文件的行数,输出 12
重定向深入讲解
一般情况下,每个 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。
$ command 2>file 希望 stderr 重定向到 file,就是说错误信息输出到file
$ command > file 2>&1 stdout 和 stderr 合并后重定向到 file
$ command < file1 >file2 将 stdin 重定向到 file1,将 stdout 重定向到 file2。
$ command > /dev/null 2>&1 屏蔽stdout 和 stderr
$ wc -l << EOF
欢迎来到
菜鸟教程
www.runoob.com
EOF # 输出结果为 3 行
cat << EOF
欢迎来到
菜鸟教程
www.runoob.com
EOF
文件包含
. filename # 注意有一空格
或
source filename
expr
1、计算字串长度
> expr length “this is a test”
14
2、抓取字串
> expr substr “this is a test” 3 5
is is
3、抓取第一个字符数字串出现的位置
> expr index "sarasara" a
2
4、整数运算
> expr 14 % 9 + 10 - 4 \* 4 / 5 注意乘号
AWK 是一种处理文本文件的语言,是一个强大的文本分析工具$1代表第一列
# 每行按空格或TAB分割,输出文本中的1、4项
$ awk '{print $1,$4}' log.txt
# 格式化输出
$ awk '{printf "%-8s %-10s\n",$1,$4}' log.txt
# 使用逗号分割.-F相当于内置变量FS, 指定分割字符
$ awk -F, '{print $1,$2}' log.txt
# 或者使用内建变量FS指定分割字符为逗号
$ awk 'BEGIN{FS=","} {print $1,$2}' log.txt
# 使用多个分隔符.先使用空格分割,然后对分割结果再使用","分割
$ awk -F '[ ,]' '{print $1,$2,$5}' log.txt
# 设置变量 -v是设置变量的意思,$1的结果都加上3
$ awk -va=3 '{print $1,$1+a}' log.txt
#过滤第一列大于2的行
$ awk '$1>2' log.txt #命令
#过滤第一列等于2的行
$ awk '$1==2 {print $1,$3}' log.txt #命令
运算符 描述
= += -= *= /= %= ^= **= 赋值
?: C条件表达式
|| 逻辑或
&& 逻辑与
~ 和 !~ 匹配正则表达式和不匹配正则表达式
< <= > >= != == 关系运算符
空格 连接
+ - 加,减
* / % 乘,除与求余
+ - ! 一元加,减和逻辑非
^ *** 求幂
++ -- 增加或减少,作为前缀或后缀
$ 字段引用
in 数组成员
内建变量
变量 描述
$n 当前记录的第n个字段,字段间由FS分隔
$0 完整的输入记录
ARGC 命令行参数的数目
ARGIND 命令行中当前文件的位置(从0开始算)
ARGV 包含命令行参数的数组
CONVFMT 数字转换格式(默认值为%.6g)ENVIRON环境变量关联数组
ERRNO 最后一个系统错误的描述
FIELDWIDTHS 字段宽度列表(用空格键分隔)
FILENAME 当前文件名
FNR 各文件分别计数的行号
FS 字段分隔符(默认是任何空格)
IGNORECASE 如果为真,则进行忽略大小写的匹配
NF 一条记录的字段的数目
NR 已经读出的记录数,就是行号,从1开始
OFMT 数字的输出格式(默认值是%.6g)
OFS 输出字段分隔符,默认值与输入字段分隔符一致。
ORS 输出记录分隔符(默认值是一个换行符)
RLENGTH 由match函数所匹配的字符串的长度
RS 记录分隔符(默认是一个换行符)
RSTART 由match函数所匹配的字符串的第一个位置
SUBSEP 数组下标分隔符(默认值是/034)