第十二章 Shell脚本编写及常见面试题(三)

简介:

本章目录:

wKiom1kuG2Dij8H6AAAuNhrwFhc739.png

12.21 FTP下载文件

1
2
3
4
5
6
7
8
9
10
11
12
13
#!/bin/bash
if  [ $ # -ne 1 ]; then
     echo  "Usage: $0 filename"
fi
dir =$( dirname  $1)
file =$( basename  $1)
ftp  -n - v  << EOF    # -n 自动登录
open  192.168.1.10
user admin adminpass
binary    # 设置ftp传输模式为二进制,避免MD5值不同或.tar.gz压缩包格式错误
cd  $ dir
get  "$file"
EOF

12.22 输入五个100数之内的字符,统计和、最小和最大

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
COUNT=1
SUM=0
MIN=0
MAX=100
while  [ $COUNT - le  5 ];  do
     read  -p  "请输入1-10个整数:"  INT
     if  [[ ! $INT =~ ^[0-9]+$ ]];  then
         echo  "输入必须是整数!"
         exit  1
     elif  [[ $INT -gt 100 ]];  then
         echo  "输入必须是100以内!"
         exit  1
     fi
     SUM=$(($SUM+$INT))
     [ $MIN -lt $INT ] && MIN=$INT
     [ $MAX -gt $INT ] && MAX=$INT
     let  COUNT++
done
echo  "SUM: $SUM"
echo  "MIN: $MIN"
echo  "MAX: $MAX"

wKioL1kuHZCQQNrOAAAloooYaNM429.png

12.23 将结果分别赋值给变量

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
方法1:
for  in  $( echo  "4 5 6" );  do
    eval  a$i=$i
done
echo  $a4 $a5 $a6
方法2:将位置参数192.168.18.1{1,2}拆分为到每个变量
num=0
for  in  $( eval  echo  $*); do    #eval将{1,2}分解为1 2
    let  num+=1
    eval  node${num}= "$i"
done
echo  $node1 $node2 $node3
# bash a.sh 192.168.18.1{1,2}
192.168.18.11 192.168.18.12
方法3:
arr=(4 5 6)
INDEX1=$( echo  ${arr[0]})
INDEX2=$( echo  ${arr[1]})
INDEX3=$( echo  ${arr[2]})

12.24 批量修改文件名

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
# touch article_{1..3}.html
# ls
article_1.html  article_2.html  article_3.html
现在想把article改为bbs:
方法1:
for  file  in  $( ls  *html);  do
     mv  $ file  bbs_${ file #*_}
     # mv $file $(echo $file |sed -r 's/.*(_.*)/bbs\1/')
     # mv $file $(echo $file |echo bbs_$(cut -d_ -f2) 
done
方法2:
for  file  in  $( find  . -maxdepth 1 -name  "*html" );  do
      mv  $ file  bbs_${ file #*_}
done
方法3:
# rename article bbs *.html

12.25 统计当前目录中以.html结尾的文件总大小

1
2
3
4
5
6
7
8
9
方法1:
# find . -name "*.html" -maxdepth 1 -exec du -b {} \; |awk '{sum+=$1}END{print sum}'
方法2:
for  size  in  $( ls  -l *.html | awk  '{print $5}' );  do
     sum =$(($ sum +$size))
done
echo  $ sum
递归统计:
# find . -name "*.html" -exec du -k {} \; |awk '{sum+=$1}END{print sum}'

12.26 扫描主机端口状态

1
2
3
4
5
6
7
8
9
10
#!/bin/bash
HOST=$1
PORT= "22 25 80 8080"
for  PORT  in  $PORT;  do
     if  echo  &> /dev/null  /dev/tcp/ $HOST/$PORT;  then
         echo  "$PORT open"
     else
         echo  "$PORT close"
     fi
done

wKiom1kuIdyj5UkEAAAONQjBJ3c657.png

12.27 Expect实现SSH免交互执行命令

需要先安装expect工具。

expect涉及用法说明:

命令描述

set可以设置超时,也可以设置变量

timeout超时等待时间,默认10s

spawn执行一个命令

expect ""匹配输出的内容

exp_continue继续执行下面匹配

\r回车

$argc统计位置参数数量

[lindex $argv 0]位置参数

puts打印字符串,类似于echo

expect{...}输入多行记录

方法1:EOF标准输出作为expect标准输入

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
#!/bin/bash
USER=root
PASS=123.com
IP=192.168.1.120
expect << EOF
set  timeout 30
spawn  ssh  $USER@$IP   
expect {
     "(yes/no)"  {send  "yes\r" ; exp_continue}
     "password:"  {send  "$PASS\r" }
}
expect  "$USER@*"   {send  "$1\r" }
expect  "$USER@*"   {send  "exit\r" }
expect eof 
EOF

方法2:

1
2
3
4
5
6
7
8
9
10
11
#!/bin/bash
USER=root
PASS=123.com
IP=192.168.1.120
expect -c "
     spawn  ssh  $USER@$IP
     expect {
         \"( yes /no )\" {send \" yes \r\"; exp_continue}
         \"password:\" {send \"$PASS\r\"; exp_continue}
         \"$USER@*\" {send \" df  -h\r  exit \r\"; exp_continue}
     }"

方法3:将expect脚本独立出来

login.exp登录文件:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
#!/usr/bin/expect 
set  ip [lindex $argv 0]
set  user [lindex $argv 1]
set  passwd  [lindex $argv 2]
set  cmd [lindex $argv 3]
if  { $argc != 4 } {
puts  "Usage: expect login.exp ip user passwd"
exit  1
}
set  timeout 30
spawn  ssh  $user@$ip
expect {
     "(yes/no)"  {send  "yes\r" ; exp_continue}
     "password:"  {send  "$passwd\r" }
}
expect  "$user@*"   {send  "$cmd\r" }
expect  "$user@*"   {send  "exit\r" }
expect eof

执行命令脚本:

1
2
3
4
5
6
7
8
#!/bin/bash
HOST_INFO=user_info
for  ip  in  $( awk  '{print $1}'  $HOST_INFO)
do
     user=$( awk  - v  I= "$ip"  'I==$1{print $2}'  $HOST_INFO)
     pass=$( awk  - v  I= "$ip"  'I==$1{print $3}'  $HOST_INFO)
     expect login.exp $ip $user $pass $1
done

SSH连接信息文件:

# cat user_info 

192.168.1.120 root 123456

12.28 批量修改服务器用户密码

旧密码SSH主机信息old_info文件:

#     ip     user    passwd    port

#--------------------------------------

192.168.18.217  root    123456     22

192.168.18.218  root    123456     22

修改密码脚本:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
#!/bin/bash
OLD_INFO=old_info
NEW_INFO=new_info
for  IP  in  $( awk  '/^[^#]/{print $1}'  $OLD_INFO);  do
     USER=$( awk  - v  I=$IP  'I==$1{print $2}'  $OLD_INFO)
     PASS=$( awk  - v  I=$IP  'I==$1{print $3}'  $OLD_INFO)
     PORT=$( awk  - v  I=$IP  'I==$1{print $4}'  $OLD_INFO)
     NEW_PASS=$(mkpasswd -l 8)
     echo  "$IP   $USER   $NEW_PASS   $PORT"  >> $NEW_INFO
     expect -c "
     spawn  ssh  -p$PORT $USER@$IP
     set  timeout 2
     expect {
         \"( yes /no )\" {send \" yes \r\";exp_continue}
         \"password:\" {send \"$PASS\r\";exp_continue}
         \"$USER@*\" {send \" echo  \'$NEW_PASS\' | passwd  --stdin $USER\r  exit \r\";exp_continue}
     }"
done

生成新密码new_info文件:

192.168.18.217  root    n8wX3mU%        22

192.168.18.218  root    c87;ZnnL        22

12.29 打印乘法口诀

1
2
3
4
5
6
7
8
9
10
方法1:
# awk 'BEGIN{for(n=0;n++<9;){for(i=0;i++<n;)printf i"x"n"="i*n" ";print ""}}'
方法2:
for  ((i=1;i<=9;i++));  do
    for  ((j=1;j<=i;j++));  do
      result=$(($i*$j))
      echo  -n  "$j*$i=$result "
    done
    echo
done

wKioL1kuHgrQ_XPxAABhsSZCEJg877.png-wh_50

12.30 getopts工具完善脚本命令行参数

getopts是一个解析脚本选项参数的工具。

命令格式:getopts optstring name [arg]

初次使用你要注意这几点:

1)脚本位置参数会与optstring中的单个字母逐个匹配,如果匹配到就赋值给name,否则赋值name为问号;

2)optstring中单个字母是一个选项,如果字母后面加冒号,表示该选项后面带参数,参数值并会赋值给OPTARG变量;

3)optstring中第一个是冒号,表示屏蔽系统错误(test.sh: illegal option -- h);

4)允许把选项放一起,例如-ab

下面写一个打印文件指定行的简单例子,用于引导你思路,扩展你的脚本选项功能:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
#!/bin/bash
while  getopts  :f:n: option;  do
     case  $option  in 
         f)
             FILE=$OPTARG
         [ ! -f $FILE ] &&  echo  "$FILE File not exist!"  &&  exit
             ;;
         n)
             sed  -n  "${OPTARG}p"  $FILE
             ;;
         ?)
             echo  "Usage: $0 -f <file_path> -n <line_number>"
             echo  "-f, --file           specified file"
             echo  "-n, --line-number    print specified line"
             exit  1
         ;;
     esac
done

wKioL1kuID3zfLJWAAAeiazGKzY694.png

思路扩展:限定脚本参数,将参数保存变量,下面调用变量继续操作。



本文转自 李振良OK 51CTO博客,原文链接:http://blog.51cto.com/lizhenliang/1930804,如需转载请自行联系原作者

相关文章
|
2月前
|
Shell
一个用于添加/删除定时任务的shell脚本
一个用于添加/删除定时任务的shell脚本
118 1
|
1月前
|
Shell Linux 测试技术
6种方法打造出色的Shell脚本
6种方法打造出色的Shell脚本
71 2
6种方法打造出色的Shell脚本
|
1月前
|
XML JSON 监控
Shell脚本要点和难点以及具体应用和优缺点介绍
Shell脚本在系统管理和自动化任务中扮演着重要角色。尽管存在调试困难、可读性差等问题,但其简洁高效、易于学习和强大的功能使其在许多场景中不可或缺。通过掌握Shell脚本的基本语法、常用命令和函数,并了解其优缺点,开发者可以编写出高效的脚本来完成各种任务,提高工作效率。希望本文能为您在Shell脚本编写和应用中提供有价值的参考和指导。
67 1
|
1月前
|
Ubuntu Shell 开发工具
ubuntu/debian shell 脚本自动配置 gitea git 仓库
这是一个自动配置 Gitea Git 仓库的 Shell 脚本,支持 Ubuntu 20+ 和 Debian 12+ 系统。脚本会创建必要的目录、下载并安装 Gitea,创建 Gitea 用户和服务,确保 Gitea 在系统启动时自动运行。用户可以选择从官方或小绿叶技术博客下载安装包。
63 2
|
2月前
|
监控 网络协议 Shell
ip和ip网段攻击拦截系统-绿叶结界防火墙系统shell脚本
这是一个名为“小绿叶技术博客扫段攻击拦截系统”的Bash脚本,用于监控和拦截TCP攻击。通过抓取网络数据包监控可疑IP,并利用iptables和firewalld防火墙规则对这些IP进行拦截。同时,该系统能够查询数据库中的白名单,确保合法IP不受影响。此外,它还具备日志记录功能,以便于后续分析和审计。
66 6
|
1月前
|
运维 监控 Shell
深入理解Linux系统下的Shell脚本编程
【10月更文挑战第24天】本文将深入浅出地介绍Linux系统中Shell脚本的基础知识和实用技巧,帮助读者从零开始学习编写Shell脚本。通过本文的学习,你将能够掌握Shell脚本的基本语法、变量使用、流程控制以及函数定义等核心概念,并学会如何将这些知识应用于实际问题解决中。文章还将展示几个实用的Shell脚本例子,以加深对知识点的理解和应用。无论你是运维人员还是软件开发者,这篇文章都将为你提供强大的Linux自动化工具。
|
2月前
|
监控 Unix Shell
shell脚本编程学习
【10月更文挑战第1天】shell脚本编程
85 12
|
2月前
|
存储 运维 监控
自动化运维:使用Shell脚本简化日常任务
【9月更文挑战第35天】在IT运维的日常工作中,重复性的任务往往消耗大量的时间。本文将介绍如何通过编写简单的Shell脚本来自动化这些日常任务,从而提升效率。我们将一起探索Shell脚本的基础语法,并通过实际案例展示如何应用这些知识来创建有用的自动化工具。无论你是新手还是有一定经验的运维人员,这篇文章都会为你提供新的视角和技巧,让你的工作更加轻松。
84 2
|
3月前
|
Shell
shell脚本变量 $name ${name}啥区别
shell脚本变量 $name ${name}啥区别
|
2月前
|
存储 Shell Linux
【Linux】shell基础,shell脚本
Shell脚本是Linux系统管理和自动化任务的重要工具,掌握其基础及进阶用法能显著提升工作效率。从简单的命令序列到复杂的逻辑控制和功能封装,Shell脚本展现了强大的灵活性和实用性。不断实践和探索,将使您更加熟练地运用Shell脚本解决各种实际问题
37 0