老男孩原题网址:http://oldboy.blog.51cto.com/2561410/1632876
写shell是一个很好玩的事儿,每次看见别人的想法,就手痒痒想用shell写出来,因为每一次写对了都会有成就感,但是写shell会成瘾,写了一个又想看看下一个,搞的nginx和dns这几天都没怎么看,于是就没有把20道题都写出来。shell答案不是死板的,但是思路基本都是一样的。本人所写的这几个都亲自测试过,均好使。
面试题4:批量创建10个系统帐号oldboy01-oldboy10并设置密码(密码为随机8位大写英文字母)。
【思路】设定随机密码的方法有很多种,这里说一种比较简单的方法。
#yum install expect,然后使用mkpasswd命令来设定密码,这个命令的基本参数如下:
-l 密码长度,默认是10;
-C 大写字母个数,默认是2;
-c 小写字母个数,默认是2;
-d 数字个数,默认是2;
-s 特殊字符,默认是1.
如果要设定一个"长度为10,大写字母3个,含有2个特殊字符的密码":#mkpasswd -l 10 -C 3 -s 2
如题要设定一个"长度为8,全部都是大写字母":#mkpasswd -l 8 -C 8 -d 0 -c 0 -s 0(要把其他的参数定位为0)
有了密码,可以写脚本了:
#!/bin/bash #writen by ChrisChan@2016-3-14 for i in $(seq 1 15); do echo user$i:$(mkpasswd -l 8 -C 2) >> useradd.txt useradd -d /usr/user$i -m user$i #使用useradd添加用户 echo $i is ok! #提示已经完成一个项目 sleep 1 #沉睡1秒 done
面试题2:使用for循环在/oldboy目录下通过随机小写10个字母加固定字符串oldboy批量创建10个html文件,例如:
#Author:ChrisChan #E-mail:[email protected] #!/bin/bash for i in $(seq 1 10); do name=$(mkpasswd -l 10 -c 10 -s 0 -d 0 -C 0) echo $name"_oldboy.html" is created! touch $name"_oldboy.html" done
面试题3:将以上文件名中的oldboy全部改成oldgirl(用for循环实现),并且html改成大写。(请用至少两种方法实现)
【思路1】先试用grep命令把所有后缀为_oldboy.html的文件全部集合,然后用for循环提取出前面随机的部分,然后使用mv 一一对应改名。
【思路2】同样先用grep将所有后缀为_oldboy.html的文件名整合到一个叫name的文件里,然后使用sed进行匹配更改,同时生成对应的“前缀_oldgirl.HTML"文件,然后再mv原有的文件到新的文件里。
这里只写"思路1":
#!/bin/bash #writen by ChrisChan@2016-3-14 file=$(ls|grep _oldboy.html|grep -v .sh) for i in $file; do first=$(echo $i|cut -d _ -f 1) mv $i $first"_oldgirl.HTML" echo $first is OK~ done
面试题5:写一个脚本,实现判断10.0.0.0/24网络里,当前在线用户的IP有哪些。
【思路】如何判断用户是否在线?ping他呗!
#!/bin/bash #writen by ChrisChan@2016-3-14 for i in $(seq 1 255); do ping 10.0.0.$i -c 2 >> /dev/null 2>&1 #无论ping到ping不到都不在前台显示 tai=$(echo $?) if [ $tai == 0 ]; then echo -e "\033[1;32m 203.90.146.$i is online \033[0m" #加重颜色显示online else echo -e "\033[1;35m 203.90.146.$i is offline \033[0m" #同上 fi done
注意,这个方法虽然直观简单,但是由于是单线程,非常非常的浪费时间,真要是ping200多个,估计要花5~6分钟。所以最好使用多线程。
那多线程怎么做?
#!/bin/bash #writen by ChrisChan@2016-3-14 for i in $(seq 1 255); do { ping 10.0.0.$i -c 2 >> /dev/null 2>&1 #无论ping到ping不到都不在前台显示 tai=$(echo $?) if [ $tai == 0 ]; then echo -e "\033[1;32m 203.90.146.$i is online \033[0m" #加重颜色显示online else echo -e "\033[1;35m 203.90.146.$i is offline \033[0m" #同上 fi }& done wait echo "all over!"
多了一个&和wait,没了。效果立竿见影,不信的话,可以#date && sh 这个脚本.sh && date,看看时间差。
实战题6:写一个脚本解决DOS***生产案例,根据web日志或者或者网络连接数,监控当某个IP并发连接数或者短时内PV达到100,即调用防火墙命令封掉对应的IP,监控频率每隔3分钟。防火墙命令为:iptables -I INPUT -s 对应的ip -j DROP。(请用至少两种方法实现!)
【思路1】nginx的nginx.conf可以设置access_log的格式,remote_addr就是客户请求地址,一般来说这个默认是在第一位。然后从这个access_log里查询ip地址。
注意,这里使用read line而不是for,因为checkip.txt的格式是“XX IP地址”的形式,如果用for的话,会按空格切块i,而read line是整条语句的输出。
#!/bin/bash writen by ChrisChan@2016-3-15 log=/usr/local/nginx/logs/checkip.txt #设定路径 function add_iptables(){ #设定一个函数叫add_iptables while read line #使用read line而不是for循环 do count=$(echo $line|awk '{print $1}') ip=$(echo $line|awk '{print $2}') if [ count -ge 100 ] then iptables -I INPUT -s $ip -j DROP echo $ip is DROPPED fi done<$log #使用<输入重定向,把line输入进去 } function main(){ while true do awk '{print $1}' access_log|sort|uniq -c >/usr/local/nginx/logs/checkip.txt add_iptables sleep 180 #每隔180秒 done } main
【思路2】使用netstat -an来查看ip链接情况,重定向到一个文件里,然后后面的写法都是一样的。
面试题10:bash for循环打印下面这句话中字母数不大于6的单词。(请用至少两种方法实现!)I am oldboy teacher and welcome to oldboy training class.
【思路】反馈字符串数字个数的方法,echo ${#变量}。比如 a="I love this game!"&& echo ${#a}
#!/bin/bash #writen by ChrisChan @ 2016-3-14 content="I am oldboy teacher and welcome to oldboy training class." for i in $content; do if [ ${#i} -le 6 ]; then echo $i fi done
判断字符串长度的方法还有一个命令,是#expr length "要检查长度的字符串",但是这两个方法的脚本是很类似的。这个应该算是思路2了,不重复写了。
但是要注意!expr length这个方法比较费时间,计算大额数字的时候,明显不如${#变量}的方法,所以还是更推荐${#变量}的方法,好记还内置函数。
面试题12:打印选择菜单,一键安装Web服务:
[root@oldboyscripts]# sh menu.sh
1.[install lamp]
2.[install lnmp]
3.[exit]
pls input the num you want:
要求:
1、当用户输入1时,输出“startinstalling lamp.”然后执行/server/scripts/lamp.sh,脚本内容输出"lampis installed"后退出脚本;
2、当用户输入2时,输出“startinstalling lnmp.”然后执行/server/scripts/lnmp.sh输出"lnmpis installed"后退出脚本;
3、当输入3时,退出当前菜单及脚本;
4、当输入任何其它字符,给出提示“Input error”后退出脚本。
5、要对执行的脚本进行相关条件判断,例如:脚本是否存在,是否可执行等。
【思路】case语句里面嵌套if语句,直线思路,注意一下fi;;收尾。
#!/bin/bash/ #writen by ChrisChan @ 2016-3-15 echo -e "1.[install lamp] \n \n 2.[install lnmp] \n \n 3.[exit]" read choose case "$choose" in"1") if [ -e /server/scripts/lamp.sh ]; then echo "startinstalling lamp." && sh /server/scripts/lamp.sh else echo "文件不存在" && exit 91 fi;; "2") if [ -e /server/scripts/lnmp.sh ]; then echo "startinstalling lnmp." && sh /server/scripts/lnmp.sh else echo "文件不存在" && exit 92 fi;; "3")echo "退出本脚本" && exit 99;;*)echo "Input Error" && exit 100;;esac
面试题15:监控web站点目录(/var/html/www)下所有文件是否被恶意篡改(文件内容被改了),如果有就打印改动的文件名(发邮件),定时任务每3分钟执行一次(10分钟时间完成)。
【思路】先备份一下web目录,起名起名叫backupwww。然后用for循环挨个diff,然后把diff结果累计输入到一个result.txt的文件里,如果这个result.txt不等于0,那么证明文件有发生了变化。脚本写完之后,crontab -e一下。
面试题17:实践抓阄题目:需要挑选5个学生,因此需要一个抓阄的程序:
要求:1、执行脚本后,想去的同学输入英文名字全拼,产生随机数01-99之间的数字,数字越大就去参加项目实践,前面已经抓到的数字,下次不能在出现相同数字。
2、第一个输入名字后,屏幕输出信息,并将名字和数字记录到文件里,程序不能退出继续等待别的学生输入。
【思路】生成随机数不难,#echo $[$RANDOM%99],但是如何要生成不重复的五个数字(5个学生抓阄)?其实很简单,先确定这个文件里的sort|uniq -c|wc-l =5,然后再random就行。
http://www.z-dig.com/the-lottery-of-shell-script-example.html写得非常详细,可以借鉴一下。
面试题18:已知下面的字符串是通过RANDOM随机数变量md5sum|cut-c 1-8截取后的结果,请破解这些字符串对应的md5sum前的RANDOM对应数字?
21029299
00205d1c
a3da1677
1f6d12dd
890684b
【思路】这属于暴力破解内容,待续~~