标签:shell习题


shell习题-统计普通用户


写个shell,看看你的Linux系统中是否有自定义用户(普通用户),若是有,一共有几个?

 

参考答案:

假设所有普通用户都是uid大于1000的

#!/bin/bash
 n=`awk -F ':' '$3>1000' /etc/passwd|wc -l`
 if [ $n -gt 0 ]
 then
     echo "There are $n common users."
 else
     echo "No common users."
 fi

shell习题-监控httpd进程


在服务器上,写一个监控脚本。

1. 每隔10s去检测一次服务器上的httpd进程数,如果大于等于500的时候,就需要自动重启一下apache服务,并检测启动是否成功?
2. 若没有正常启动还需再一次启动,最大不成功数超过5次则需要理解发邮件通知管理员,并且以后不需要再检测!
3. 如果启动成功后,1分钟后再次检测httpd进程数,若正常则重复之前操作(每隔10s检测一次),若还是大于等于500,那放弃重启并需要发邮件给管理员,然后自动退出该脚本。假设其中发邮件脚本为之前咱们使用的mail.py

 

参考答案:

#!/bin/bash

check_service()
{
    n=0
    for i in `seq 1 5`
    do
    /usr/local/apache2/bin/apachectl restart 2>/tmp/apache.err
    if [ $? -ne 0 ]
    then
        n=$[$n+1]
    else
        break
    fi
    done

    if [ $n -eq 5 ]
    then
        ##下面的mail.py参考https://coding.net/u/aminglinux/p/aminglinux-book/git/blob/master/D22Z/mail.py
        python mai.py "123@qq.com" "httpd service down" `cat /tmp/apache.err`
        exit
    fi
}   


while :
do
    t_n=`ps -C httpd --no-heading |wc -l`
    if [ $t_n -ge 500 ]
    then
        /usr/local/apache2/bin/apachectl restart
        if [ $? -ne 0 ]
        then
            check_service
        fi
        sleep 60
        t_n=`ps -C httpd --no-heading |wc -l`
        if [ $t_n -ge 500 ]
        then
            python mai.py "123@qq.com" "httpd service somth wrong" "the httpd process is budy."
            exit
        fi
    fi
    sleep 10
done

shell习题-批量创建用户并设置密码


用shell脚本实现如下需求:
添加user_00 – user_09 10个用户,并且给他们设置一个随机密码,密码要求10位包含大小写字母以及数字,注意需要把每个用户的密码记录到一个日志文件里。

提示: 
1. 随机密码使用命令 mkpasswd
2. 在脚本中给用户设置密码,可以使用echo 然后管道passwd命令

 

参考答案

#!/bin/bash
for i in `seq -w 00 09`
do
useradd user_$i
p=`mkpasswd -s 0 -l 10`
echo “user_$i $p” >>/tmp/user0_9.pass
echo $p |passwd –stdin user_$i
done


shell习题-输入数字执行对应命令


写一个脚本实现如下功能:  输入一个数字,然后运行对应的一个命令。显示命令如下:*cmd meau**  1—date 2–ls 3–who 4–pwd
当输入1时,会运行date, 输入2时运行ls, 依此类推。

 

参考答案:

#! /bin/bash

echo “*cmd meau**  1—date 2–ls 3–who 4–pwd”

read -p “please input a number 1-4: ” n

case $n in

1)

date

;;

2)

ls

;;

3)

who

;;

4)

pwd

;;

*)

echo “Please input a number: 1-4”

;;

esac


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