分类:shell习题


shell习题 – 查找字母数小于6的单词


用shell打印下面这句话中字母数小于6的单词。
Bash also interprets a number of multi-character options.

 

参考答案:

#!/bin/bash

for s in Bash also interprets a number of multi-character options

do

n=`echo $s|wc -c`

if [ $n -lt 6 ]

then echo $s

fi

done


shell习题-删除文本中的字母


要求:

把一个文本文档的前5行中包含字母的行删除掉,同时把6到10行中的全部字母删除掉。

 

参考答案:

假设文本名字叫做1.txt,并且文本行数大于10,脚本如下

#!/bin/bash

##先获取该文本的行数

nu=`wc -l 1.txt |awk ‘{print $1}’`

##对前5行进程处理

for i in `seq 1 5`

do

##使用sed把每一行的内容赋值给变量

l=`sed -n “$i”p 1.txt`

##用grep 判定是否匹配字母,-v取反,-q不输出内容

if echo $l |grep -vq ‘[a-zA-Z]’

then

echo $l

fi

done

##对6-10行做删除字母处理

for i in `seq 6 10`

do

l=`sed -n “$i”p 1.txt`

echo $l|sed ‘s/[a-zA-Z]//g’

done

##剩余的直接输出

for i in `seq 11 $nu`

do

sed -n “$i”p 1.txt

done

##若想把更改内容写入到1.txt,还需要把以上内容重定向到一个文本中,然后删除1.txt,再把刚刚重定向的文件更名为1.txt


shell习题-自动重启php-fpm服务


服务器上跑的是LNMP环境,近期总是有502现象。502为网站访问的状态码,200正常,502错误是nginx最为普通的错误状态码。由于502只是暂时的,并且只要一重启php-fpm服务则502消失,但不重启的话,则会一直持续很长时间。所以有必要写一个监控脚本,监控访问日志的状态码,一旦发生502,则自动重启一下php-fpm。

我们设定:
1. access_log  /data/log/access.log
2. 脚本死循环,每10s检测一次(假设每10s钟的日志条数为300左右)
3. 重启php-fpm的方法是  /etc/init.d/php-fpm restart

参考答案:

#! /bin/bash

log=/data/log/access.log

N=10

while :; do

##因为10秒钟大概产生300条日志

tail -n 300 $log > /tmp/log

n_502=`grep -c ‘ 502″‘ /tmp/log`

if [ $n_502 -ge $N ]; then

##记录系统的状态

top -bn1 >/tmp/`date +%H%M%S`-top.log

vmstat 1 5 >/tmp/`date +%H%M%S`-vm.log

/etc/init.d/php-fpm restart 2>/dev/null

##重启php-fpm服务后,应先暂缓1分钟,而后继续每隔10s检测一次

sleep 60

fi

sleep 10

done


shell习题-备份数据库


需求:

设计一个shell脚本来备份数据库,首先在本地服务器上保存一份数据,然后再远程拷贝一份,本地保存一周的数据,远程保存一个月。

假定,我们知道mysql root账号的密码,要备份的库为discuz,本地备份目录为/bak/mysql, 远程服务器ip为192.168.123.30,远程提供了一个rsync服务,备份的地址是 192.168.123.30::backup  . 写完脚本后,需要加入到cron中,每天凌晨3点执行。

 

参考答案:

#! /bin/bash

### backup mysql data

### Writen by Aming.

PATH=/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin:/usr/local/mysql/bin

d1=`data +%w`

d2=`date +%d`

pass=”your_mysql_password”

bakdir=/bak/mysql

r_bakdir=192.168.123.30::backup

exec 1>/var/log/mysqlbak.log 2>/var/log/mysqlbak.log

echo “mysql backup begin at `date +”%F %T”`.”

mysqldump -uroot -p$pass –default-character-set=gbk discuz >$bakdir/$d1.sql

rsync -az $bakdir/$d1.sql $r_bakdir/$d2.sql

echo “mysql backup end at `date +”%F %T”`.”

然后加入cron

0 3 * * * /bin/bash /usr/local/sbin/mysqlbak.sh


shell习题-监控80端口


需求:

写一个脚本,判断本机的80端口(假如服务为httpd)是否开启着,如果开启着什么都不做,如果发现端口不存在,那么重启一下httpd服务,并发邮件通知你自己。脚本写好后,可以每一分钟执行一次,也可以写一个死循环的脚本,30s检测一次。

 

参考答案

#! /bin/bash
mail=123@123.com
if netstat -lnp |grep ‘:80’ |grep -q ‘LISTEN’; then
exit
else
/usr/local/apache2/bin/apachectl restart >/dev/null 2> /dev/null
python mail.py $mail “check_80” “The 80 port is down.”
n=`ps aux |grep httpd|grep -cv grep`
if [ $n -eq 0 ]; then
/usr/local/apache2/bin/apachectl start 2>/tmp/apache_start.err
fi
if [ -s /tmp/apache_start.err ]; then
python mail.py  $mail  ‘apache_start_error’   `cat /tmp/apache_start.err`
fi
fi


shell习题-批量更改文件名


需求:

  1. 找到/123目录下所有后缀名为.txt的文件
  2. 批量修改.txt为.txt.bak
  3. 把所有.bak文件打包压缩为123.tar.gz
  4. 批量还原文件的名字,即把增加的.bak再删除

参考答案

#!/bin/bash

##查找txt文件

find /123 -type f -name “*.txt” > /tmp/txt.list

##批量修改文件名

for f in `cat /tmp/txt.list`

do

mv $f $f.bak

done

##创建一个目录,为了避免目录已经存在,所以要加一个复杂的后缀名

d=`date +%y%m%d%H%M%S`

mkdir /tmp/123_$d

##把.bak文件拷贝到/tmp/123_$d

for f in `cat /tmp/txt.list`

do

cp $f.bak /tmp/123_$d

done

##打包压缩

cd /tmp/

tar czf 123.tar.gz 123_$d/

##还原

for f in `cat /tmp/txt.list`

do

mv $f.bak $f

done


shell练习题-设计监控脚本


设计一个脚本,监控远程的一台机器(假设ip为123.23.11.21)的存活状态,当发现宕机时发一封邮件给你自己。

提示:
1. 你可以使用ping命令   ping -c10 123.23.11.21
2. 发邮件脚本可以参考 https://coding.net/u/aminglinux/p/aminglinux-book/git/blob/master/D22Z/mail.py
3. 脚本可以搞成死循环,每隔30s检测一次

 

参考答案:

#!/bin/bash

ip=123.23.11.21
ma=abc@139.com

while 1

do
ping -c10 $ip >/dev/null 2>/dev/null
if [ $? != “0” ];then
python /usr/local/sbin/mail.py $ma “$ip down” “$ip is down”

#假设mail.py已经编写并设置好了
fi
sleep 30
done


shell练习题-统计内存使用


需求:

写一个脚本计算一下linux系统所有进程占用内存大小的和。(提示,使用ps或者top命令)

 

参考答案:

#! /bin/bash

sum=0

for mem in `ps aux |awk ‘{print $6}’ |grep -v ‘RSS’ `

do

sum=$[$sum+$mem]

done

echo “The total memory is $sum””k”

也可以使用awk 一条命令计算:

ps aux | grep -v ‘RSS TTY’ |awk ‘{(sum=sum+$6)};END{print sum}’


shell练习题-统计日志


有日志1.log,内容如下:

日志片段:

112.111.12.248 – [25/Sep/2013:16:08:31 +0800]formula-x.haotui.com “/seccode.php?update=0.5593110133088248″ 200″http://formula-x.haotui.com/registerbbs.php” “Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1;)”

61.147.76.51 – [25/Sep/2013:16:08:31 +0800]xyzdiy.5d6d.com “/attachment.php?aid=4554&k=9ce51e2c376bc861603c7689d97c04a1&t=1334564048&fid=9&sid=zgohwYoLZq2qPW233ZIRsJiUeu22XqE8f49jY9mouRSoE71″ 301″http://xyzdiy.5d6d.com/thread-1435-1-23.html” “Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1; .NET CLR 1.1.4322; .NET CLR 2.0.50727)”

要求: 统计出每个IP的访问量有多少?

 

参考答案:

awk ‘{print $1}’ 1.log |sort -n|uniq -c |sort -n


shell练习题-每日生成一个文件


要求:

请按照这样的日期格式(xxxx-xx-xx)每日生成一个文件,例如今天生成的文件为)2017-07-05.log, 并且把磁盘的使用情况写到到这个文件中,(不用考虑cron,仅仅写脚本即可)

 

参考答案:

#! /bin/bash
d=`date +%F`
logfile=$d.log
df -h > $logfile