您的位置:首页 > 运维架构 > Shell

shell 脚本-符号-基础语法

2016-08-22 16:50 507 查看
为了便于识别 建议 以.sh 结尾
shell脚本 可以放上所有的命令行的指令(源于bash)
shell脚本 是 解释型语言 c/c++ 是编译型语言
下面用到的 shell代码sh_1.sh
#!/bin/bash
cd ../
ls
pwd
解释型脚本的执行过程:
script.sh 是文本文件,根本没有代码段和 _start 函数 , exec 怎么执行。
解释:
Shell会fork 一个子进程并调用 exec执行 ./script.sh这个程序,exec 系统调用应该把子进程的代码
段替换成./script.sh程序的代码段 ,并从它的 _start开始执行。然而 script.sh是个文本文件,根 本
没有代码段和 _start函数 ,怎么办呢 ?其实 exec还有另外一种机制 ,如果要执行的是一个文本文
件, 并且第一行用 Shebang 指定了解释器 ,则用 解释器程序的代码段替换当前进程 ,并且从解释
器 的_start开始执行 ,而这个文本文件被当作命令行参数传给解释器 。因此, 执行上述脚本相
当于执 行程序



1. 交互 Shell(bash)fork/exec一个子 Shell(sh)用于执行脚本 ,父进程 bash等待子进程 sh终止。
2. sh读取脚本中的 cd ..命令 ,调用相应的函数 执行内建命令 ,改变当前工作目录为上一级
目 录。
3. sh读取脚本中的 ls命令 ,fork/exec这个程序 ,列出当前工作目录下的文件 ,sh等待 ls终止。
4. ls终止后 ,sh继续执行 ,读到脚本文件末尾 ,sh终止。
5. sh终止后 ,bash继续执行 ,打印提示符等待用户输入。
如果将命令行下输入的命令用 ()括号括起来 ,那么也会 fork出一个子 Shell执行小括号中的命令 ,
一 行中可以输入由分号 ;隔开的多个命令 ,比如 : $ (cd ..;ls -l)
和上面两种方法执行 Shell脚本的效果是相同的 ,cd ..命令改变的是子 Shell的 PWD,而不会影响
到 交互式Shell。
chmod + x script.sh方式
sh文件中,cd ..命令改变的是子Shell的PWD,而不会影响到 交互式Shell1(对于文件sh, 交互式shell1会先创建一个子shell2,子shell2会再创建一个shell3,shell3遇见cd命令 会直接由shell2执行,改变的是shell2的路径,shell1的路径不变)。
然而source ./script.sh 和 . ./script.sh 方式
则有不同的效果,cd ..命令是直接在交互式Shell1下执行的,改变交互式Shell的PWD
对于php 。。。脚本语言 执行也是 解释器这个流程
sh_1.sh代码:
#!/bin/bash cd ../ ls pwd

[bozi@localhost 1_shell]$ /bin/bash sh_1.sh // shell脚本中 cd 直接在子bash中运行 整个过程父进程 不参与
1_shell
/home/bozi/linux_test/shell
[bozi@localhost 1_shell]$ pwd
/home/bozi/linux_test/shell/1_shell // 所以 不影响父进程 的路径 还在子进程的路径中
[bozi@localhost 1_shell]$ cd .. // cd内置命令 shell 自己直接亲自执行(不创建子进程) 影响自己的路径
[bozi@localhost shell]$ pwd
/home/bozi/linux_test/shell // 直接执行 影响了 跑到上级目录
路径

[bozi@localhost shell]$ ll
总用量 4
drwxrwxr-x. 2 bozi bozi 4096 8月 14 10:18 1_shell
[bozi@localhost shell]$ cd 1_shell/
[bozi@localhost 1_shell]$ ll
总用量 4
-rw-rw-r--. 1 bozi bozi 298 8月 14 10:18 sh_1.sh
source 与(. 命令一样)
例子:
source 不创建子bash,遇见bash中有内建命令 cd 交互式shell(父进程自己直接执行) 所以 退出时 父进程的路径 变了 影响父进程
[bozi@localhost 1_shell]$ ll
总用量 4
-rw-rw-r--. 1 bozi bozi 298 8月 14 10:18 sh_1.sh
[bozi@localhost 1_shell]$ source sh_1.sh
1_shell
/home/bozi/linux_test/shell
[bozi@localhost shell]$ pwd
/home/bozi/linux_test/shell
.命令 效果相同
[bozi@localhost 1_shell]$ . sh_1.sh
1_shell
/home/bozi/linux_test/shell
[bozi@localhost shell]$ pwd
/home/bozi/linux_test/shell
set 显示 本地变量 和 环境变量(本地变量只存在当前进程中)
env 只显示环境变量 (环境变量 可以传递给 子进程 父子进程共享)
变量
shell 中 所有变量varname都是字符串 ,且都是全局本地变量 没有int ,float等类型 显示变量 用 $varname或 ${varname}
如果变量不存在 shell显示空串
变量拼接 用{}花括号 如: ${varname}aaa $varname"aaa"
演示代码:
[bozi@localhost 2_shell]$ SHELL=1
[bozi@localhost 2_shell]$ echo $SHELL
1
[bozi@localhost 2_shell]$ echo $SHELLabc
[bozi@localhost 2_shell]$ echo ${SHELL}
1
[bozi@localhost 2_shell]$ echo ${SHELL}abc
1abc
unset 取消一个变量
[root@localhost 2_shell]# v=a
[root@localhost 2_shell]# echo $v
a
[root@localhost 2_shell]# unset $v
[root@localhost 2_shell]#
显示本shell的pid
[root@localhost 2_shell]# echo $$
6699
read
-p 提示符 -t 等待的秒数
[bozi@localhost 2_shell]$ read -p '>>' -t 10 arg
>>nihao
[bozi@localhost 2_shell]$ echo $arg
nihao
数组变量
var[1]="small min"
var[2]="big min"
var[3]="nice min"
echo "${var[1]}, ${var[2]},${var[3]}"
[bozi@localhost 2_shell]$ ./1_test.sh
small min, big min,nice min
test指令
两个整数之间的判定
-eq 相等(equal)
-ne 不等(not equal)
-gt 大于 (greaater than)
-lt 小于 (less than)
-ge 大于等于 (greater than or equal)
-le 小于等于 (less than or equal)
例子:
[bozi@localhost 2_shell]$ test 1 -lt 2; echo $?
0                ---------------》【真】
[bozi@localhost 2_shell]$ test 5 -lt 2; echo $?
1
判定字符串的数据
test -z string [string 为空  返回true]
test -n string [string 为空  返回false]
test str1 = str2 [str1=str2 回传true]
test str1 != str2 [str1 与str2相等返回 false]
例子:
[bozi@localhost 2_shell]$ test -z "";echo $?
0
[bozi@localhost 2_shell]$ test -z "-";echo $?
1
[bozi@localhost 2_shell]$ test -n "";echo $?
1
[bozi@localhost 2_shell]$ test -n "-";echo $?
0
[bozi@localhost 2_shell]$ test "nihao" = "hello" ; echo $?
1
[bozi@localhost 2_shell]$ test "nihao" == "hello" ; echo $?
1
[bozi@localhost 2_shell]$ test "nihao" != "hello" ; echo $?
0
error
[bozi@localhost 2_shell]$ test "nihao"=="hello" ; echo $?
0 ----》“==两边少空格 结果 不正确”
test扩展:
当要检测系统上面某些文件或者是相关的属性时,利用test这个命令来工作真是好用得不得了,如检查/dmtsai是否存在时,使用:test –e /dmtsai上面的执行结果并不会显示任何信息,但最后可以通过$?或&&及||来显示整个结果。test –e /dmtsai && echo “exist” ||echo “Not exist”最终结果可以显示exist还是not exist。-e是测试一个东西存在不存在。常用的测试命令如下:
测试的标志
代表意义
关于某个文件名的“文件类型”判断,如test – e filename表示存在否
-e
该文件名是否存在
-f
该文件名是否存在且为文件(file)
-d
该文件名是否存在且为目录(directory)
-b
该文件名是否存在且为一个block device设备
-c
该文件名是否存在且为一个character device设备
-S
该文件名是否存在且为一个Socket文件
-p
该文件名是否存在且为一个FIFO(pipe)文件
-L
该文件名是否存在且为一个连接文件
关于文件的权限检测,如test –r filename表示可读否(但root权限常有例外)
-r
检测该文件名是否存在且具有“可读”的权限
-w
检测该文件名是否存在且具有“可写”的权限
-x
检测该文件名是否存在且具有“可执行”的权限
-u
检测该文件名是否存在且具有“SUID”的属性
-g
检测该文件名是否存在且具有“SGID“的属性
-k
检测该文件名是否存在且具有“Sticky bit”的属性
-s
检测该文件名是否存在且具有“非空白文件”
两个文件之间的比较,如test file1 –nt file2
-nt
(newer than)判断file1是否比file2新
-ot
(older than)判断file1是否比file2旧
-ef
判断file1与file2是否为同一文件,可用在判断hard link的判定上。主要意义在于判定两个文件是否均指向同一个inode
关于两个整数之间的判定吗,如test n1 –eq n2
-eq
两数值相等(equal)
-ne
两数值不等(not equal)
-gt
N1大于n2(greate than)
-lt
N1小于n2(less than)
-ge
N1大于等于n2(greater than or equal)
-le
N1小于等于n2(less than or equal)
判定字符串的数据
test –z string
判定字符串是否为0,若string为空字符串,则为true
test –n string
判定字符串是否非为0,若string为空字符串,则为false
test str1 = str2
判定str1是否等于str2,若相等,则回传true
test str1 != str2
判定str1是否不等于str2,若相等,则回传false
多重条件判定,若test –r filename –a –x filename
-a
两个条件同时成立!如test –r file –a –x file,则file同时具有r与x权限时,才回传true
-o
任何一个条件成立!如test –r file –o –x file,则file具有r或x权限时,就可回传true

反向状态,如test ! –x file,但file不具有x时,回传true
多重条件判定
-a [先当于 与&&]
-o 【或】
! 【非】
例子:
[bozi@localhost 2_shell]$ test 1 -eq 1 -a 2 -lt 5; echo $?
0
[bozi@localhost 2_shell]$ test 1 -eq 1 -a 5 -lt 2; echo $?
1
判断符号[]
[bozi@localhost 2_shell]$ [ "" ==  "HOME" ];echo $?
1
[bozi@localhost 2_shell]$ [ "" !=  "HOME" ];echo $?
0

error
[bozi@localhost 2_shell]$ [ ""!="HOME" ];echo $?
0
[bozi@localhost 2_shell]$ [ ""=="HOME" ];echo $?
0
注意 空格不能少 否则出错:
[bozi@localhost 2_shell]$ [ 空格"" 空格== 空格"HOME" 空格];echo $?
建议:
1 在中括号[]中的每个组件用 空格 隔开
2 在中括号内的变量, 最好用双引号括起来
3 在中括号里面的常量最好用单引号或 双引号括起来
2的一个错误例子:
[bozi@localhost 2_shell]$ name="hello world"
[bozi@localhost 2_shell]$ [ $name == "hello" ]
bash: [: too many arguments   --------------------------》  太多参数 本来是两个参数比较 因为name变量的字符串中间有空格 不加“”
解析为 [ hello world == "hello" ]   肯定参数太多
[bozi@localhost 2_shell]$ [ "$name" == "hello" ]
[bozi@localhost 2_shell]$ echo $?
1
[bozi@localhost 2_shell]$ [ "$name" != "hello" ]
[bozi@localhost 2_shell]$ echo $?
0
小练习:
read -p "please inoput (Y/N):" yn
[ "$yn" == "Y" -o "$yn" == "y" ] && echo "ok, continue" && exit 0
[ "$yn" == "N" -o "$yn" == "n" ] && echo "oh, interrupt" && exit 0
echo "I dont know what your chonice is " && exit 0
运行:
[bozi@localhost 2_shell]$ ./1_test.sh
please inoput (Y/N):y
ok, continue
[bozi@localhost 2_shell]$ ./1_test.sh
please inoput (Y/N):n
oh, interrupt
[bozi@localhost 2_shell]$ ./1_test.sh
please inoput (Y/N):
I dont know what your chonice is
shell中&&和||的使用方法
&&运算符:

command1 && command2

&&左边的命令(命令1)返回真(即返回0,成功被执行)后,&&右边的命令(命令2)才能够被执行;换句话说,“如果这个命令执行成功&&那么执行这个命令”。
语法格式如下:

command1 && command2 [&& command3 ...]

1 命令之间使用 && 连接,实现逻辑与的功能。
2 只有在 && 左边的命令返回真(命令返回值 $? == 0),&& 右边的命令才会被执行。
3 只要有一个命令返回假(命令返回值 $? == 1),后面的命令就不会被执行。
||运算符:
command1 || command2

||则与&&相反。如果||左边的命令(命令1)未执行成功,那么就执行||右边的命令(命令2);或者换句话说,“如果这个命令执行失败了||那么就执行这个命令。
1 命令之间使用 || 连接,实现逻辑或的功能。
2 只有在 || 左边的命令返回假(命令返回值 $? == 1),|| 右边的命令才会被执行。这和 c 语言中的逻辑或语法功能相同,即实现短路逻辑或操作。
3 只要有一个命令返回真(命令返回值 $? == 0),后面的命令就不会被执行。
&& || 与 -a -o
区别
-a -o 连接的是两个表达式 即 测试条件
而 && 与 || 连接的是两条 命令
代码:
val=10
str="hello"
test $val -eq 10 -a "$str" == "hello"
echo $?
test $val -eq 10 && test "$str" == "hello"
echo $?
运行:
[bozi@localhost 2_shell]$ ./12_test.sh
0
0
默认变量 $0 $1 $2 ...
例子:$ ./1_test.sh nihao sunshine
$0 $1 $2
$# :代表后接的参数的个数, 如上例子为2
$@:代表"$1" "$2" ... 之意, 每个变量独立的用双括号括起来
$*: 代表“$1c$2c$3c$4” c为分隔符号,默认是空格键
$@ 和 $*还是有所不同的 一般用 $@
代码:
#! /bin/bash
echo "\$0 is $0"
echo "\$1 is $1"
echo "\$2 is $2"
echo "\$# is $#"
echo "\$@ is $@"
echo "\$* is $*"
运行:
[bozi@localhost 2_shell]$ ./1_test.sh
$0 is ./1_test.sh
$1 is
$2 is
$# is 0
$@ is
$* is
[bozi@localhost 2_shell]$ ./1_test.sh 1 2 3 4
$0 is ./1_test.sh
$1 is 1
$2 is 2
$# is 4
$@ is 1 2 3 4
$* is 1 2 3 4
代码:
echo "your whole parameter is ===>$0"
echo "total parameter numbers is ===>$#"
[ "$#" -lt 2 ]&&echo "the numbers of paramater is less than 2. stop here"\
&& exit 0;
echo "your whole parameter is ===>$@"
echo "the 1st parameter ===>$1"
echo "the 2nd parameter ===> $2 "
运行:
[bozi@localhost 2_shell]$ ./1_test.sh
your whole parameter is ===>./1_test.sh
total parameter numbers is ===>0
the numbers of paramater is less than 2. stop here
[bozi@localhost 2_shell]$ ./1_test.sh nihao sunshine
your whole parameter is ===>./1_test.sh
total parameter numbers is ===>2
your whole parameter is ===>nihao sunshine
the 1st parameter ===>nihao
the 2nd parameter ===> sunshine
运行:
[bozi@localhost 1_shell]$ cat sh_1.sh
#!/bin/bash
myint=10
echo '\$ \\ $myint \"'
echo '###############'
echo "\$ \\ $myint \""
[bozi@localhost 2_shell]$ ./1_test.sh your whole parameter is ===>./1_test.sh total parameter numbers is ===>0 the numbers of paramater is less than 2. stop here [bozi@localhost 2_shell]$ ./1_test.sh nihao sunshine your whole parameter is ===>./1_test.sh total parameter numbers is ===>2 your whole parameter is ===>nihao sunshine the 1st parameter ===>nihao the 2nd parameter ===> sunshine
[bozi@localhost 2_shell]$ ./1_test.sh nihao sunshine here
your whole parameter is ===>./1_test.sh
total parameter numbers is ===>3
your whole parameter is ===>nihao sunshine here
the 1st parameter ===>nihao
the 2nd parameter ===> sunshine
shift:造成参数号码偏移,移除前num个参数
shift+num num 及num之前的参数全部移除 num之后的参数从$1开始
如:
echo $@
echo "arg count $#"
shift 1
echo "after shift 1"
echo $@
echo "arg count $#"
shift 3
echo "after shift 3"
echo $@
echo "arg count $#"
运行:
[bozi@localhost 2_shell]$ ./1_test.sh one two three four five six seven
one two three four five six seven
arg count 7
after shift 1
two three four five six seven
arg count 6
after shift 3
five six seven
arg count 3
条件判断式
1 if ... then
if [条件判断式];then
条件成立执行
fi
&&代表AND
|| 代表 or
代码:
#!/bin/bash
read -p "please input (Y/N):" yn
if [ "X$yn" == "XY" ] || [ "X$yn" == "Xy" ];then
echo "ok, continue"
exit 0
fi
if [ "X$yn" == "XN" ] || [ "X$yn" == "Xn" ];then
echo "oh, interupt!"
exit 0
fi
运行:
please input (Y/N):y
ok, continue
[bozi@localhost 2_shell]$ ./2_test.sh
please input (Y/N):Y
ok, continue
[bozi@localhost 2_shell]$ ./2_test.sh
please input (Y/N):N
oh, interupt!
[bozi@localhost 2_shell]$ ./2_test.sh
please input (Y/N):n
oh, interupt!
多重复杂条件判断式
if [条件判断式];then
条件成立
else
...
if
if [条件判断式1];then
...
elif [条件判断式2];then
...
else
...
if
代码:
read -p "please input (Y/N):"  yn
if [ "x$yn" == "xY" ] || [ "x$yn" == "xy" ]; then
echo "ok, continue"
elif [ "x$yn" == "xN" ] || [ "x$yn" == "xn" ]; then
echo "oh, interupt"
else
echo "I dont know what your choice is"
fi
运行:
[bozi@localhost 2_shell]$ vim 2_test.sh
[bozi@localhost 2_shell]$ ./2_test.sh
please input (Y/N):y
ok, continue
[bozi@localhost 2_shell]$ ./2_test.sh
please input (Y/N):n
oh, interupt
[bozi@localhost 2_shell]$ ./2_test.sh
please input (Y/N):s
I dont know what your choice is
代码:
testing=$(netstat -tuln |grep ':80' )
if [ "X$testing" != "X" ];then
echo "WWW is running in your system ."
fi
testing=$(netstat -tuln | grep ':22')
if [ "X$testing" != "X" ]; then
echo "SSH is running in your system."
fi
testing=$(netstat -tuln | grep ':21')
if [ "X$testing" != "X" ]; then
echo "FTP is running in your system."
fi
testing=$(netstat -tuln | grep ':25')
if [ "X$tesing" != "X" ]; then
echo "Mail is running in your system."
fi
运行:
[bozi@localhost 2_shell]$ ./3_text.sh
Now, I will detect your linux server's services!
The wwww, ftp, ssh , and mail will be detect !
SSH is running in your system.
利用case ...... esac判断
case $变量名称 in
”第1个变量名称“)
程序段
;;
”第2个变量名称“)
程序段
;; # ;; 相当于break

*) #*)相当于default
程序段
;;
esac
代码:
#!/bin/bash
case $1 in
"one")
echo "your choice is ONE"
;;
"two")
echo "your choice is TWO"
;;
"three")
echo "your choice is THREE"
;;
*)
echo "Usage $0 {one|two|three}"
;;
esac
运行:
[bozi@localhost 2_shell]$ ./4_test.sh
Usage ./4_test.sh {one|two|three}
[bozi@localhost 2_shell]$ ./4_test.sh one
your choice is ONE
[bozi@localhost 2_shell]$ ./4_test.sh two
your choice is TWO
[bozi@localhost 2_shell]$ ./4_test.sh three
your choice is THREE
函数 function
函数的参数 也是$1$2$.... 但是函数体里面的$1$2...与函数外面的 是相对独立的
函数返回值: return后 用 $? 来接收 【返回值】(缺陷 返回257 是 1 范围0-255)
代码:
#!/bin/bash
function printit()             # 也可以不加关键字function  即  printit(){...}   也是可以的
{
echo -n "Your choice is "  #加上-n表示不断行继续在一行显示
}
case $1 in
"one")
printit;echo $1 |tr 'a-z' 'A-Z' #大小写转化
;;
"two")
printit;echo $1 |tr 'a-z' 'A-Z' #大小写转化
;;
"three")
printit;echo $1 |tr 'a-z' 'A-Z' #大小写转化
;;
*)
echo "Usage $0 {one|two|three}"
;;
esac
运行:
[bozi@localhost 2_shell]$ ./5_test.sh one
Your choice is ONE
[bozi@localhost 2_shell]$ ./5_test.sh
Usage ./5_test.sh {one|two|three}
[bozi@localhost 2_shell]$ ./5_test.sh one
Your choice is ONE
[bozi@localhost 2_shell]$ ./5_test.sh two
Your choice is TWO
[bozi@localhost 2_shell]$ ./5_test.sh three
Your choice is THREE
经典的fork炸弹 【!!!立马宕机】
.() { .|.& }; .             #递归调用  后台执行
类似的 function a() { a|a }; a
防范措施 ulimit -Hu num 限定用户最多num个进程
-------------------------------------------------
以下程序段就是由Jaromil所作的在类UNIX系统的shell环境下触发fork炸弹的shell脚本代码,总共只用了13个字符(包括空格):
:(){ :|:& };:
注解如下:
:() # 定义函数,函数名为":",即每当输入":"时就会自动调用{}内代码
{ # ":"函数开始标识
: # 用递归方式调用":"函数本身
| # 并用管道(pipe)将其输出引至...
: # 另一次递归调用的":"函数
# 综上,":|:"表示的即是每次调用函数":"的时候就会生成两份拷贝【二倍指数增长】
& # 调用间脱钩,以使最初的":"函数被杀死后为其所调用的两个":"函数还能继续执行
} # ":"函数结束标识
; # ":"函数定义结束后将要进行的操作...
: # 调用":"函数,"引爆"fork炸弹
其中函数名“:”只是简化的一例,实际实现时可以随意设定,一个较易理解(将函数名替换为“forkbomb”)的版本如下:
forkbomb(){ forkbomb|forkbomb &} ; forkbomb
Windows下则可以批处理命令如下实现:
%0|%0
POSIX标准下的C与C++的实现:
#include <unistd.h>int main(){while(1) fork();return0;}
Perl语言的实现:
fork while fork
-------------------------------------------------
循环
while do done, until do done(不定循环)
代码:
#!/bin/bash
while [ "X$yn" != "Xyes" -a "X$yn" != "XYES" ]
do
read -p "please input yes/YES to stop this program:" yn
done
echo "ok, you input the correct answer."
运行:
[bozi@localhost 2_shell]$ chmod u+x 6_test.sh
[bozi@localhost 2_shell]$ ./6_test.sh
please input yes/YES to stop this program:e
please input yes/YES to stop this program:e
please input yes/YES to stop this program:yes
ok, you input the correct answer.
until代码:
sum=0
i=0
until [ $i -gt 100 ]
do
if (( i%2==0 ));then
let sum+=i
fi:
let i++
done
echo $sum
运行:
[bozi@localhost 2_shell]$ ./6_test.sh
2550
代码:
s=0
i=0
while [ "$i" != "100" ]
do
i=$(($i+1))
s=$(($s+$i))
done
echo "sum of 1+2+3+...+100 is : $s"
运行:
[bozi@localhost 2_shell]$ ./7_test.sh
sum of 1+2+3+...+100 is : 5050
for ... do... done(固定循环)
for var in con1 con2 con3 ...
do
程序段
done
代码:
for animal in dog cat elephant
do
echo "there are ${animal}s..."
done
运行:
[bozi@localhost 2_shell]$ ./8_test.sh
there are dogs...
there are cats...
there are elephants...
代码:
users=$(cut -d':' -f1 /etc/passwd)
for username in $users
do
id $username
done
运行:
[bozi@localhost 2_shell]$ ./9_test.sh
uid=0(root) gid=0(root) 组=0(root)
uid=1(bin) gid=1(bin) 组=1(bin),2(daemon),3(sys)
uid=2(daemon) gid=2(daemon) 组=2(daemon),1(bin),4(adm),7(lp)
uid=3(adm) gid=4(adm) 组=4(adm),3(sys)
uid=4(lp) gid=7(lp) 组=7(lp)
uid=5(sync) gid=0(root) 组=0(root)
测ip代码:
read -p "enter like 192.168.1->" network
for sitenu in $(seq 1 100)
do
ping -c 1 -w 1 ${network}.${sitenu} &>/dev/null && result=0 || result=1
if [ "$result" == 0 ];then
echo "server ${network}.${sitenu} is up."
else
echo "server ${network}.${sitenu} is down"
fi
done
运行:
[bozi@localhost 2_shell]$ ./9_test.sh
enter like 192.168.1->192.168.174
server 192.168.174.1 is up.
server 192.168.174.2 is up.
server 192.168.174.3 is down
server 192.168.174.4 is down
server 192.168.174.5 is down
server 192.168.174.6 is down
server 192.168.174.7 is down
代码:
#!/bin/bash
read -p "please input a diretory:" dir
if [ "$dir" == "" -o ! -d "$dir" ];then
echo "then $dir is not exist in your system."
exit 1
fi
filelist=$(ls $dir)
for filename in $filelist
do
perm=""
test -r "$dir/$filename" && perm="$perm readable"
test -w "$dir/$filename" && perm="$perm writable"
test -x "$dir/$filename" && perm="$perm executable"
echo "the file $dir/$filename's permission is $perm"
done
运行:
[bozi@localhost 2_shell]$ ./10_test.sh
please input a diretory:../1_shell
the file ../1_shell/2's permission is  readable writable
the file ../1_shell/3's permission is  readable writable
the file ../1_shell/sh_1.sh's permission is  readable writable
the file ../1_shell/sh_2.sh's permission is  readable writable executable

for ....do...done 的数值处理
for ((初始值; 限制值;执行步阶))
do
程序段
done
这种写法 适合在数值运算中
代码:
read -p "please input a number . I will count for 1+2+3+...+your_input" num
s=0
for ((i=1; i <= $num; i++))
do
((s += i))
done
echo "the result of '1+2+3+...+$num 'is $s"
运行:
[bozi@localhost 2_shell]$ ./11_test.sh
please input a number . I will count for 1+2+3+...+your_input10
the result of '1+2+3+...+10 'is 55
代码:
#!/bin/bash
for i in {A..b}
do
echo "val is: $i"
done
运行:
[bozi@localhost 2_shell]$ sh 14_test.sh
val is: A
val is: B
val is: C
val is: D
val is: E
val is: F
val is: G
val is: H
val is: I
val is: J
val is: K
val is: L
val is: M
val is: N
val is: O
val is: P
val is: Q
val is: R
val is: S
val is: T
val is: U
val is: V
val is: W
val is: X
val is: Y
val is: Z
val is: [
val is:
val is: ]
val is: ^
val is: _
val is: `
val is: a
val is: b
shell script 的 追踪 与 debug
-n : 不执行script,仅查询语法错误
-v: 在执行script之前,现将script的内容输出到屏幕上
-x: 将使用到的script内容显示到屏幕上
代码:
检查是否有语法问题 没有问题就什么也不输出
运行:
[bozi@localhost 2_shell]$ sh -n 12_test.sh
[bozi@localhost 2_shell]$
代码:
将使用到的代码显示出来
运行:
[bozi@localhost 2_shell]$ sh -x 12_test.sh
+ val=10
+ str=hello
+ test 10 -eq 10 -a hello == hello
+ echo 0
0
+ test 10 -eq 10
+ test hello == hello
+ echo 0
0
一是在命令行提供参数
$ sh -x ./script.sh
二是在脚本开头提供参数
#! /bin/sh -x
第三种方法是在脚本中用 set命令启用或禁用参数
set -x和set +x 分别表示启用和禁用 -x参数 ,这样可以只对脚本中的某一段进行跟踪调试。
代码:
val=10
str="hello"
set -x
test $val -eq 10 -a "$str" == "hello"
echo $?
set +x
test $val -eq 10 && test "$str" == "hello"
echo $?
运行:
[bozi@localhost 2_shell]$ sh -x 12_test.sh
+ val=10
+ str=hello
+ set -x
+ test 10 -eq 10 -a hello == hello
+ echo 0
0
+ set +x
0
:是一个特殊的命令,称为空命令 ,该命令不做任何事, Exit Status总是真。此外 ,也可以执 行
/bin/true或 /bin/false得到真或假的 Exit Status。

文件名替换
这些用于匹配的字符称为通配符 (Wildcard),具体如下 :
通配符 * : 匹配 0个或多个 任意字符
? : 匹配 一个任意 字符
[若干字符 ] : 匹配 方括号中任意一个字符的一次出现
运行:
[bozi@localhost 1_shell]$ ls *
2  3  sh_1.sh  sh_2.sh
[bozi@localhost 1_shell]$ ls sh_?.sh
sh_1.sh  sh_2.sh
[bozi@localhost 1_shell]$ ls [s]h*
sh_1.sh  sh_2.sh
注意, Globbing 所匹配的文件名是由 Shell展开的 ,也就是说在参数还没传给程序之前已经展开
了 ,不是ls展开的
命令代换:
由反引号``括起来的也是一条命令 ,Shell先执行该命令 ,然后将输出结果立刻代换到当前命令行
中。
[bozi@localhost 1_shell]$ DATE='date'
[bozi@localhost 1_shell]$ echo $DATE
date
[bozi@localhost 1_shell]$ DATE=`date`
[bozi@localhost 1_shell]$ echo $DATE
2016年 08月 21日 星期日 11:05:13 CST
例如定义一个变量存放 date命令的输出 :
命令代换也可以用 $()表示 : $ DATE=$(date)
[bozi@localhost 1_shell]$ DATE=$(date)
[bozi@localhost 1_shell]$ echo $DATE
2016年 08月 21日 星期日 11:05:56 CST


算术代换 :$(())
用于算术计算 ,$(())中的 Shell变量取值将转换成整数 ,例如 :
[bozi@localhost 1_shell]$ v=3
[bozi@localhost 1_shell]$ echo $((v+3))
6
$(())中只能用+-*/ 和()运算符 ,并且只能做整数运算。

(( )) 中间支持c语言的写法
[bozi@localhost 1_shell]$ i=1;(( i++));echo $i
2
[bozi@localhost 1_shell]$ i=1;(( i+=5));echo $i
6
[bozi@localhost 1_shell]$ i=1;(( i>0));echo $?
0
[bozi@localhost 1_shell]$ i=1;(( i<0));echo $?
1
shell进度条小程序
代码:
#!/bin/bash
function proc()
{
i=0;
str=''
arr=('|' '/' '-' '\\')
index=0
while [ $i -le 100 ]
#while ((i <= 100))
do
printf "[%-100s][%d%%][%c]\r" "$str" "$i" "${arr[$index]}"
#这里不用像c语言一样fflush刷新输出缓冲区 因为 printf是子进程执行
# 子进程退出自动刷新
str=${str}'#'
sleep 0.1
let i++
let index++
let index%=4
#((index%=4))
done
printf "\n"
}
function main()
{
proc
}
main
运行:
[bozi@localhost 4_shell]$ sh proc.sh
#########################################################################                          ][74%][-]
读取文件数据 计算最大和最小值、平均值
代码:
#找 文件中的最大值和 最小值
max=0
min=0
count=0
sum=0
while read line
do
if [ "$count" -eq "0" ];then
max=$line
min=$line
let count++
let sum+=$line
continue
fi
[ "$max" -lt "$line" ] && max=$line
[ "$min" -gt "$line" ] && min=$line
let sum+=$line
let count++
done <file
echo $max
echo $min
echo "ibase=10 ;scale=2; $sum/$count" | bc
数据:
[bozi@localhost 4_shell]$ cat file
1
2
3
4
5
6
7
8
9
运行:
[bozi@localhost 4_shell]$ sh ./5_shell.sh
9
1
5.00
建议数字计算 用(()) 不用[ ] []对于空数据报错
如数据第四行空
file
[bozi@localhost 4_shell]$ cat file
1
2
3
5
6
7
8
9
(())结果

if (( line <  9));then
[bozi@localhost 4_shell]$ vim test.sh
[bozi@localhost 4_shell]$ sh  ./test.sh
1
2
3
5
6
7
8
error
[]结果 if [ "$line" -lt "9" ];then
[bozi@localhost 4_shell]$ sh  ./test.sh
1
2
3
./test.sh: line 11: ((: <  9 : syntax error: operand expected (error token is "<  9 ")
error
5
6
7
8
error
数组
1 数组的定义
一对()表示数组 数组元素用 空格 符号作为分隔符
[bozi@localhost 4_shell]$ a=(1 2 3 4 5)
[bozi@localhost 4_shell]$ echo $a
1
[bozi@localhost 4_shell]$ echo ${a[*]}
1 2 3 4 5
[bozi@localhost 4_shell]$ str=('q' "nihao" 'hello')
[bozi@localhost 4_shell]$ echo $str
q
[bozi@localhost 4_shell]$ echo ${str[@]}
q nihao hello
2 数组的读取与赋值(下标从0开始)
赋值:
[bozi@localhost 4_shell]$ a[10]=10
读取:
[bozi@localhost 4_shell]$ echo ${a[*]}
1 2 3 4 5 10
[bozi@localhost 4_shell]$ echo ${a[@]}
1 2 3 4 5 10
得到长度:
得到整个数组长度
[bozi@localhost 4_shell]$ echo ${#a[*]}
6
[bozi@localhost 4_shell]$ echo ${#a[@]}
6
得到数组某个元素的长度
[bozi@localhost 4_shell]$ a=(1 2 3 4 5 10)
[bozi@localhost 4_shell]$ echo ${a[@]} 1 2 3 4 5 10
[bozi@localhost 4_shell]$ echo ${#a[5]}
2
[bozi@localhost 4_shell]$ echo ${#a[0]}
1
删除:
[bozi@localhost 4_shell]$ echo ${a[@]}
1 2 3 4 5 10
#删除某个元素
[bozi@localhost 4_shell]$ unset a[2]
[bozi@localhost 4_shell]$ echo ${a[@]}
1 2 4 5 10
#删除整个数组
[bozi@localhost 4_shell]$ unset a
[bozi@localhost 4_shell]$ echo ${a[@]}
[bozi@localhost 4_shell]$
3.特殊使用:
分片:
$[数组名{@或*]起始位置:长度} 从起始位置开始 切长度个元素
返回的是字符串 中间用空格隔开
所以如果加上"()" , 将的到切片的数组。
[bozi@localhost 4_shell]$ echo ${a[@]}
1 2 3 4 5 6
[bozi@localhost 4_shell]$ echo ${a[@]:0:3}
1 2 3
[bozi@localhost 4_shell]$ echo ${a[@]:2:3}
3 4 5
[bozi@localhost 4_shell]$ echo ${a[@]:2:5}
3 4 5 6
[bozi@localhost 4_shell]$ echo ${a[@]:1:5}
2 3 4 5 6
加上"()" , 将的到切片的数组。
[bozi@localhost 4_shell]$ echo ${a[@]:1:5}
2 3 4 5 6
[bozi@localhost 4_shell]$ newa=(${a[@]:1:5})
[bozi@localhost 4_shell]$ echo ${newa[@]}
2 3 4 5 6
替换:
调用方法是: ${数组名[@或*]/查找字符/替换字符} 该操作不会改变原先的内容, 如果需要修改, 可以像上面一样 加() 生成一个新的数组
[bozi@localhost 4_shell]$ a=(1 2 3 4 5 6)
[bozi@localhost 4_shell]$ echo ${a[@]}
1 2 3 4 5 6
[bozi@localhost 4_shell]$ echo ${a[@]/3/100}
1 2 100 4 5 6
[bozi@localhost 4_shell]$ echo ${a[@]}
1 2 3 4 5 6
原内容 未改变
要改变原内容 加()
[bozi@localhost 4_shell]$ a=(${a[@]/3/100})
[bozi@localhost 4_shell]$ echo ${a[@]}
1 2 100 4 5 6
[bozi@localhost 4_shell]$ str=("nihao" hello)
[bozi@localhost 4_shell]$ echo ${str[@]}
nihao hello
[bozi@localhost 4_shell]$ echo ${str[@]/ni/wo}
wohao hello
[bozi@localhost 4_shell]$ str=("nihao" "hello" "here")
[bozi@localhost 4_shell]$ echo ${str[@]}
nihao hello here
[bozi@localhost 4_shell]$ echo ${str[@]/he/the}
nihao thello there
遍历数组:
arr=(aaa bbb ccc ddd)
num=${#arr[@]}
for ((i = 0; i < num; i++))
{
echo ${arr[i]}
}
[bozi@localhost 4_shell]$ sh arr.sh
aaa
bbb
ccc
ddd
特殊
arr=(aaa bbb ccc ddd)
arr[50]="ffff"
num=${#arr[*]}
for ((i = 0; i < num; i++))
{
echo ${arr[i]}
}
结果 多了一个元素arr[50] 读也是读五个 只不过是从前向后读 没有读到arr[50] 读到空
这里${#arr[*]}和 ${#arr[@]}都是5
[bozi@localhost 4_shell]$ sh arr.sh
aaa
bbb
ccc
ddd
[bozi@localhost 4_shell]$ cat arr.sh
下面这种方式解决读到空的问题 没有读到后面的有效数字
代码:
arr=(aaa bbb ccc ddd)
arr[50]="ffff"
num=${#arr[*]}
for var in ${arr[*]}
#for var in ${arr[@]}       这两个效果一样 都能访问到定义了的数据

{
echo $var
}
运行:
[bozi@localhost 4_shell]$ sh arr.sh
aaa
bbb
ccc
ddd
ffff
[bozi@localhost 4_shell]$ for var in ${arr[*]}; do echo $var;let i++;done
a
b
c
hello
[bozi@localhost 4_shell]$ arr[100]="iam100"
[bozi@localhost 4_shell]$ for var in ${arr[*]}; do echo $var;let i++;done
a
b
c
hello
iam100
[bozi@localhost 4_shell]$ [bozi@localhost 4_shell]$ for var in ${arr[*]}; do echo $var;let i++;done
a
b
c
hello
[bozi@localhost 4_shell]$ arr[100]="iam100"
[bozi@localhost 4_shell]$ for var in ${arr[*]}; do echo $var;let i++;done
a
b
c
hello
iam100
[bozi@localhost 4_shell]$ for var in ${arr[@]}; do echo $var;let i++;done
a
b
c
hello
iam100
[bozi@localhost 4_shell]$ i=0;while ((i < ${#arr[*]})); do echo ${arr[i]};let i++;done
a
b
c
hello
[bozi@localhost 4_shell]$ i=0;while ((i < ${#arr[@]})); do echo ${arr[i]};let i++;done
a
b
c
hello
[bozi@localhost 4_shell]$
${arr[@]}; do echo $var;let i++;done
a
b
c
hello
iam100
[bozi@localhost 4_shell]$ i=0;while ((i < ${#arr[*]})); do echo ${arr[i]};let i++;done
a
b
c
hello
[bozi@localhost 4_shell]$ i=0;while ((i < ${#arr[@]})); do echo ${arr[i]};let i++;done
a
b
c
hello
[bozi@localhost 4_shell]$
从这个 例子可以看出
for var in种方式访问的是 有效元素的个数
for ((i = 0; i < len;i++)) 是从前向后找 ,找len个就停止,后面隔着的有些有效的数据就没有输出
count=0; while [ $count -le 100] touch test${count};
数学运算(( ))中变量直接取得是 变量的值
v1=1
v2=2
((v3 = v1 + v2)) // (($v3 = $v1 + $v2)) 也可以
echo $(v3)
echo $(arr[*]) 显示所有的元素 (空的丢弃)
echo $(arr[@]) 显示所有的元素
echo $(#arr[@]) 显示 已经定义的元素的个数 如 arr
遍历数组
for i in $(arr[@])
do
echo $(arr[i])
done
单引号 双引号
单引号 对转义、特殊的变量 都不会 进行处理 原样输出
双引号 会处理
[bozi@localhost 1_shell]$ /bin/bash sh_1.sh
10
\$ \\ $myint \"
###############
$ \ 10 "
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  Linux