shell脚本中直接 ssh 后写文件 echo到一个文件_shell不允许输入空字符


第8章 shell脚本编程

一. shell简介

当命令不在命令行中执行,而是从一个文件中执行时,该文件就称为 Shell 脚本。shell脚本是纯文本文件,以行为单位逐行执行。

相当于一种命令翻译器。

二. shell脚本的编写

2.1脚本的创建

使用文本编辑器(或touch)创建脚本文件

语法:touch a.sh

2.2脚本的执行

将脚本添加可执行权限

语法:chmod +x a.sh

执行脚本命令:

(1)./a.sh

(2)sh a.sh

(3)source a.sh //不需要x权限

2.3脚本的编写

编辑脚本:vim a.sh

#!/bin/bash //系统以bash解释器执行脚本

cd /etc/

pwd //命令语句

三. shell功能

3.1重定向

重定向输出:将命令的执行结果覆盖到目标文件

语法:df > a.txt //将磁盘信息覆盖到a.txt文件

重定向追加:uname -p >> a.txt //将处理器类型追加到a.txt

重定向输入:

vim pass.txt

123456

passwd --stdin jerry < pass.txt //标准输入

tail -2 /etc/shadow //查看用户密码信息后两行

3.2管道符号

将左侧的命令输出结果,作为右侧命令的处理对象

cat a.txt |grep “123” //筛选出含字符串“123”的行

grep "bash$" /etc/passwd | awk -F: '{ print $1,$7 }' //-F:后要求有空格

例:提取根分区/的磁盘使用情况

df -hT //查看磁盘使用情况-h显示更易读-T显示文件系统类型

df -hT | grep "/$" | awk '{print $6}'

四. shell变量

shell变量用来存放系统和用户需要使用的特定参数(值),而且这些参数可以根据用户的设定或系统环境的变化而相应变化。

4.1自定义变量

4.1.1定义一个新的变量(输入)

格式:变量名=变量值

4.1.2查看变量的值(输出)

格式:echo $变量名

echo命令的选项:

-n:取消自动换行

-e:使用转义字符,把字符串中某些字符当成特殊字符处理

:制表符

:换行符

b:删除前一个字符

f:换行但光标仍旧停留在原来的位置

:光标移至行首,但不换行

赋值时使用引号

双引号:允许通过$符号引用其他变量值

单引号:禁止引用其他变量值,$视为普通字符

反撇号:命令替换,提取命令执行后的输出结果

4.1.3从键盘输入内容为变量赋值(提示输入)

格式: read [-p "提示信息"] 变量名

4.2整数变量运算

Shell变量的数值运算多用于脚本程序的过程控制,只能进行整数运算,不支持小数 运算。整数的运算主要通过内部命令Expr进行。

格式:expr 变量1 运算符 变量2 [运算符 变量3]... //注意空格

常用运算符:加(+)、减(-)、乘(*)、除(/)、取模(%)

例:设置X(值为35)、Y(值为16)两个变量,并进行加、减、乘、除、取模运算

X=35

Y=16

expr $X + $Y

expr $X - $Y

expr $X * $Y //乘法符号在shell中有其它含义,运算时需要转意符号

expr $X / $Y

expr $X % $Y

4.3环境变量

由系统提前创建,用来设置用户的工作环境

配置文件:/etc/profile 、~/.bash_profile

env :查看当前工作环境下的环境变量。

这表示为 $n,n为1~9之间的数字

例:编写一个加法运算的小脚本a.sh,用来计算两个整数的和。需要计算的两个整数在执行脚本时以位置变量的形式提供。

#vim a.sh

#!/bin/bash

SUM=`expr $1 + $2`

echo "$1 + $2 = $SUM"

#chmod +x a.sh

#./a.sh 12 24 //$1为12、$2为24

4.4预定义变量

是由Bash程序预先定义好的一类特殊变量,使用"$"符号和另一个符号组合表示。

$# :命令行中位置变量的个数

$* :所有位置变量的内容

$? :表示上一条命令执行后返回的状态,当返回状态值为0时表示执行正常,非0值表示执行异常或出错

$0 :当前执行的进程/程序名

例:编写一个备份小脚本

#vim mybak.sh

#!/bin/bash

TARFILE=beifen-`date +%s`.tgz

tar zcf $TARFILE $* &> /dev/null

echo "已执行$0脚本,"

echo "共完成$#个对象的备份"

echo "具体内容包括:$*"

#chmod +x mybak.sh

#./mybak.sh /boot/grub

#./mybak.sh /etc/passwd /etc/shadow

五. 测试脚本命令

5.1条件测试

测试特定的表达式是否成立,当条件成立时,测试语句的返回值为0,否则为其他数值

格式1:test 条件表达式

格式2:[ 条件表达式 ] //注意空格,中括号与表达式之间有空格

执行条件测试操作以后,通过预定义变量"$?"获取返回状态值。

5.2文件测试

根据给定的路径名称,判断对应的是文件还是目录,或者判断文件是否可读、 可写、可执行等。

用法:[ 操作符 文件或目录 ]

-d:测试是否为目录(Directory)

-e:测试目录或文件是否存在(Exist)

-f:测试是否为文件(File)

-r:测试当前用户是否有权读取(Read)

-w:测试当前用户是否有权写入(Write)

-x:测试是否设置有可执行权限(Excute)

例:

[ -d /media/cdrom ]

echo $?

0 //返回0表示条件成立,返回1表示条件不成立

使用逻辑与"&&"和echo命令一起更直观的显示结果

[ -d /media/cdrom ] $$ echo "yes"

yes

5.3整数值比较

根据给定的两个整数值,判断第一个与第二个数的关系

用法:[ 整数1 操作符 整数2]

-eq:等于(Equal)

-ne:不等于(Not Equal)

-gt:大于(Greater Than)

-lt:小于(Lesser Than)

-le:小于或等于(Lesser or Equal)

-ge:大于或等于(Greater or Equal)

例:判断当前已登录的用户数,超出五个时输出"Too many"

Unum=`who | wc -l` //查看当前已登录用户数

[ $Unum -gt 5 ] && echo "Too many."

5.4字符串比较

通常用来检查用户输入、系统环境等是否满足条件

用法1:[ 字符串1 = 字符串2] [ 字符串1 != 字符串2 ]

用法2:[ -z 字符串 ] //检查是否为空

通用 :[ 字符串1 操作符 字符串2 ]

例1:判断当前系统语言,不是"http://en.US"时输出"Not http://en.US "

echo $LANG

[ $LANG != "http://en.US" ] && echo "Not http://en.US"

例2:用户需输入yes或no来确认操作

read -p "是否覆盖现有文件(yes/no)?" ACK

[ $ACK = "yes" ] && echo "覆盖"

read -p "是否覆盖现有文件(yes/no)?" ACK

[ $ACK = "no" ] $$ echo "不覆盖"

5.5逻辑测试

判断两个或多个条件之间的关系

用法1:[ 表达式1 ] 操作符 [ 表达式2 ]

用法2:命令1 操作符 命令2

&& :逻辑与,“而且”,使用test命令时可改为"-a"

|| :逻辑或,“或者”,使用test命令时可改为"-o"

! :逻辑否,“不”

例:测试当前系统内核版本是否符合要求

uname -r

Mnum=$(uname -r | awk -F. '{print $1}')

Snum=$(uname -r | awk -F. '{print $2}')

[ $Mnum -eq 2 ] && [ $Snum -gt 4 ] && echo "符合要求"

六. if条件语句

根据复杂程度:单分支的if语句,双分支的if语句,多分支的if语句

6.1单分支if语句

只有条件成立时才会执行一条相应的代码

用法:

if 条件测试

then 命令序列

fi

例:判断挂载点目录是否存在,若不存在则新建此目录

vim chkmountdir.sh

#!/bin/bash

MOUNT_DIR="/media/cdrom/"

if [ ! -d $MOUNT_DIR ]

then

mkdir -p $MOUNT_DIR

fi

chmod +x chkmountdir.sh //给chkmountdir.sh执行的权限

./chkmountdir.sh

6.2双分支if语句

用法:

if 条件测试操作

then 命令序列1

else 命令序列2

fi

例:检查目标主机是否能连通,显示相应信息

vim pinghost.sh

#!/bin/bash

ping -c 3 -i 0.2 -w 3 $1 &> /dev/null //命令执行时的消息都不看

if [ $? -eq 0] //判断前一条命令的返回状态

then

echo "Host $1 is up."

else

echo "Host $1 is down."

fi

chmod +x pinghost.sh

./pinghost.sh 192.168.4.11

6.3多分支if语句

用法:

if 条件测试操作1

then 命令序列1

elif 条件测试操作2

then 命令序列2

else

命令序列3

fi

例:判断分数范围,分出优秀、合格、不合格三档

vim gradediv.sh

#!/bin/bash

read -p "请输入您的分数(0-100):" GRADE

if [ $GRADE -ge 85 ] && [ &GRADE -le 100 ]

then

echo "$GRADE 分!优秀"

elif [ $GRADE -ge 70 ] && [ &GRADE -le 84 ]

then

echo "$GRADE 分,合格"

else

echo "$GRADE分?不合格"

fi

chmod +x gradediv.sh

./gradediv.sh

七. for循环语句

读取不同的变量,用来逐个执行同一组命令

用法:

for 变量名 in 取值列表

do

命令序列

done

例1:根据姓名列表批量添加用户

vim /root/users.txt //用作测试列表文件
jdy
ttl
tcc
vim uaddfor.sh
#!/bin/bash
ULIST=$(cat /root/users.txt)
for UNAME in $ULIST //从列表文件读取用户名
do
useradd $UNAME
echo "123456" | passwd --stdin $UNAME &> /dev/null
//通过管道指定密码字串
done
chmod +x uaddfor.sh
./uaddfor.sh
tail -3 /etc/passwd

例2:根据IP地址列表检查主机状态

vim /root/ipadds.txt
192.168.4.11
192.168.4.100
192.168.4.120
vim chkhosts.sh
#!/bin/bash
HLIST=$(cat /root/ipadds.txt)
for IP in $HLIST
do
ping -c 3 -i 0.2 -W 3 $IP &> /dev/null
if [ $? -eq 0 ] //嵌套if语句判断连通性
then
echo "Host $IP is up."
else
echo "Host $IP is down."
fi
done
chmod +x chkhosts.sh
./chkhosts.sh

for循环语句非常适合用于列表对象无规律,且列表来源已经固定。而对于要求控制循环次数,操作对象按数字顺序编号、按特定条件执行重复操作等情况,则更适合while循环语句。

八. while循环语句

重复测试某个条件,只要条件成立则反复执行

用法:

while 条件测试操作

do

命令序列

done

例:批量添加规律编号的用户

#vim uaddwhile.sh

#!/bin/bash

PREFIX="stu"

i=1

while [ $i -le 20 ]

do

useradd ${[PREFIX}$i

echo "123456" | passwd --stdin ${PREFIX}$i &> /dev/null

let i++ //序号递增,避免死循环

done

chmod +x uaddwhile.sh

./uaddwhile.sh

grep "stu" /etc/passwd | tail -3

例:猜价格游戏,要求由脚本预先生成一个随机的价格数目(0-99)作为实际价格,判断用户猜测的价格是否高出或低于实际价格,给出相应的提示后再次要求用户猜测,直到用户猜中实际价格为止,输出用户共猜测的次数、实际价格。

#vim pricegame.sh

#!/bin/bash

PRICE=$(expr $RANDOM % 1000) //输入price一个随机数

TIMES=0

echo "商品实际价格范围为0-999,猜猜看是多少?"

while true //循环条件:true

do

read -p "请输入你猜测的价格数目:" INT

let TIMES++

if [ $INT -eq $PRICE ] ; then

echo "恭喜你答对了,实际价格是$PRICE"

echo "你总共猜测了$TIMES次"

exit 0

elif [ $INT -gt $PRICE ] ; then

echo "太高了!"

else

echo "太低了!"

fi

done

chmod +x pricegame.sh

./pricegame.sh

九. case分支语句

针对变量的不同取值,分别执行不同的命令序列

用法:

case 变量值 in

模式1)

命令序列1

;;

模式2)

命令序列2

;;

......

*)

默认命令序列

esac

例:检查用户输入的字符类型

提示用户从键盘输入一个字符,通过case语句判断该字符是否为字母、数字或者其他控制字符,并给出相应的提示信息。

#vim hitkey.sh

#!/bin/bash

read -p "请输入一个字符,并按Enter键确认:" KEY

case "$KEY" in

[a-z] || [A-Z])

echo "你输入的是 字母."

;;

[0-9])

echo "你输入的是 数字."

;;

*)

echo "你输入的是 空格、功能键或其他控制字符."

esac

chmod -x hitkey.sh

./hitkey.sh

十. sed、awk语句

10.1 Sed简介

sed是stream editor(流编辑器)的缩写,是一种在线编辑器,它一次处理一行内容。sed是非交互式的编辑器。

处理时,把当前处理的行存储在临时缓冲区中,称为“模式空间”(pattern space),接着用sed命令处理缓冲区中的内容,处理完成后,把缓冲区的内容送往屏幕。接着处理下一行,这样不断重复,直到文件末尾。文件内容并没有改变,除非你使用重定向存储输出。

10.2 Sed命令用法

格式:sed+[选项]+‘命令’+文件名

常用选项:

-n :取消默认输入

-e :多项编辑

-i :直接修改读取的档案内容,而不是由屏幕输出

常用命令:

a :当前行后添加行

i :当前行前添加行

c :用字符串替换行

d :删除行

p :打印行

s :用一个字符串替换另一个

w :将所选的行写入文件

10.3 Sed定址

用于‘命令’之前,决定对哪些行进行编辑。

如果没有指定地址,sed将处理输入文件的所有行。

1.地址是一个数字,则表示行号;是“$"符号,则表示最后一行。

2.地址可以指定范围,当需指定范围时使用逗号(,)隔开。

10.4 Sed用法

1.p命令

格式:sed -n ‘行号p’ 文件名

2.d命令

格式:sed -n ‘行号d’ 文件名

3.s命令

格式:sed -n ‘行号s/a/b/g’文件名

将文件中指定行里所有的a字符替换为b字符,g为全局变量

4.i命令

格式:sed -i ‘行号3d’文件名

直接修改文件内容,不打印在屏幕上

5.e命令

格式:sed -e ‘3d’ -e ‘4p’ 文件名

先删除第三行,再打印第四行字符

10.5 awk简介

awk是一个强大的文本分析工具,相对于grep的查找,sed的编辑,awk在其对数据分析并生成报告时,显得尤为强大。

简单来说awk就是把文件逐行的读入,以空格为默认分隔符将每行切片,切开的部分再进行各种分析处理。

10.6 awk用法

1.格式:awk ' pattern {action} '

pattern常指关系表达式,例如:

awk‘/a/’file——显示文件中包含a的行

awk‘$2>10’file——显示第二列中数字大于10的行

2. 分隔提取:awk –F“分隔符”‘{print $1}’

例如:

awk –F.‘{print $1, print $2}’表示以点为分隔符将每一行的前二个字段,分行输出

10.7 awk实例

判定输入ip的有效性:

vim a.sh

#!/bin/bash

read -p “请输入ip:” A

B=`echo $A |awk -F. “{print $1}”`

if [ $B -le 0 ] || [ $B -gt 255 ]

then echo “请输入正确的IP”

else

echo “ok”

fi

chmod +x a.sh

sh a.sh

十一. shell实例

例1:编写一个系统服务脚本

vim myprog

#!/bin/bash

case "$1" in

start)

echo -n "正在启动sleep服务..."

if sleep 7200 &

then

echo "OK"

fi

;;

stop)

echo -n "正在停止sleep服务..."

pkill "sleep" &> /dev/sull

echo "OK"

;;

status)

if pgerp "sleep" &> /dev/sull ;

then

echo "sleep服务已经启动."

else

echo "sleep服务已经停止."

fi

;;

restart)

$0 stop

$0 start

;;

esac

chmod +x myprog

./myprog start

例2: shell打印九九乘法表

使用for循环

for i in $(seq 1 9)

do

for j in $(seq 1 $i)

do

echo -n “$j * $i” = $[i * j] “ ”

done

echo

done

//例中$[i * j]可写为:$((i * j))、$[$j*$i]、$(( $j * $i ))、`expr $i * $j`

//修改:第二个do下加一行 let “temp = i * j”,再把$[i * j]改为:$temp

while 反向的乘法表:

i=9

j=1

while ((i>=1))

do

while ((j<=i))

do

echo -n “$j * $i”=$[$j * $i] “ ”

let j++

done

len j=1

let i--

echo “”

done

----------------------------------------------end------------------------------------------------