在案例中学习shell编程

时间:2021-08-08 04:25:07

1. 编写脚本/root/bin/systeminfo.sh, 显示当前主机系统信息,包括主机名,IPv4 地址,

操作系统版本,内核版本,CPU 型号,内存大小,硬盘大小。

分析:

在案例中学习shell编程

 

解释:

此题是一堆命令的堆叠,首先我们很容易在命令行通过命令知道题目中的一些结果。

主机名:hostname命令就可以了。
IPV4地址:ifconfig命令是用来查看ip地址的,只是有很多多余内容,那么我们可以通过正则表达式,

                  以及cut命令即可截取。

操作系统版本:使用cat /etc/redhat-relase即可查得。
内核版本:uname -r 可以了。
CPU型号:有一个文件是关于cpu的即/proc/cpuinfo,这里是要cpu的型号,我们使用grep

                  结合正则截取含有mode的就很容易得到想要的结果。

内存大小:也有一个文件关于内存的即/proc/meminfo接下来就是截取的问题了。
硬盘大小:使用命令 fdisk -l ,可以查看到详细信息,接下来使用grep和cut得到想要的列。

在这里我们需要知道如何调用一个命令可以使用 $(命令)或·命令·这两种方式 。

得到内存的大小,还可以使用free -m ,计算内存的大小时,如果需要转换单位,那就得

知道如何进行变量之间的运算了,后面会提到 。

看似简单的题,更让我们学会:

                          1.写脚本规范很重要

                          2.使用变量可以让脚本变得比较有调理

                          3.熟练使用正则可以让我们事半功倍 

答:

#!/bin/bash
#------------------------
#Filename:systeminfo.sh
#Revision:1.0
#Date:2017-08-08
#Description:systeminfo
#------------------------
Ipv4=`ifconfig eth0 |grep "\<inet\>"|tr -s ' ' :|cut -d : -f 4`
Sysmode=`cat /etc/redhat-release`
Kermode=`uname -r`
CPU=`lscpu|grep "Model name"|cut -d : -f 2|tr -s ' '`
Ferr=$[$(cat /proc/meminfo |head -1|grep -o "[[:digit:]]\+")/1024]
Fdisk=`fdisk -l|head -n 2|tail -1 |cut -d , -f 1 |cut -d: -f2`
echo "系统基本信息"
echo "主机名:$(hostname)"
echo "IPV4地址:$Ipv4"
echo "系统版本:$Sysmode"
echo "内核版本:$Kermode"
echo "CPU型号:$CPU"
echo "内存大小:${Ferr}MB"
echo "硬盘大小:$Fdisk"
unset Ipv4 Sysmode Kermode CPU Ferr Fdisk

#########################################################

最后讲一下关于变量之间的计算的,在这里只讲三种方法:

方法一 :let  var=算数表达式  eg.  let  count=2+4     echo $count  就变成6了,推荐使用此发

方法二:var=$[算数表达式 ]  eg.  count=$[2*3]   echo $count  变成6了

方法三:var=((算数表达式))和上例差不多。

2. 编写脚本/root/bin/sumid.sh,计算/etc/passwd文件中的第10个用户和第20用户的ID之和.

分析:

在案例中学习shell编程

 

解释:

既然是计算UID之和,那么我们首先必须知道如何得出UID,方法并不难,使用cut以冒号为分隔符,就可以得到我

们想要的UID,剩下的事情就是如何计算两个数的和,在上一题中我们也教大家方法了,这次我们来详细讨论如

何计算两个变量之和,还有我们需要注意的是,本题中是第10个用户和第20个用户UID之和,如何使该题的通用

性增强呢,也就是说,当我们输入数字(代表第几个用户),就能计算之和,这是后我们就需要考虑使用位置

变量,当然也会涉及正则表达式的内容。举个例子,加入我们输入的不是10和20,而是一个未知数,只能用

$1和$2替代,得想办法筛选出$1行和$2行的UID并计算,我们可以使用cat -n 显示行号,然后利用正则筛选,

用cut截取,下面我们来看看具体实现方法。

答:

#!/bin/bash
#--------------------
#Filename:sumid.sh
#Revision:2.0
#Description:UID sum
#--------------------
read -p "please you input two arg:" number1 number2
root1=$(cat /etc/passwd | head -n $number1 |tail -1 | cut -d: -f3)
root2=$(cat /etc/passwd | head -n $number2 |tail -1 | cut -d: -f3)
let sumid=root1+root2
echo "第$number1个用户和第$number2个用户UID之和为:$sumid"
unset number1 number2 root1 root2

----------------------------------------------------------------------------------------------

3. 编写脚本/root/bin/argsnumsh,接受一个文件路径作为参数;如果参数个数小于1,则提示用户

“至少应该给一个参数”,并立即退出;如果参数个数不小于1,则显示第一个参数所指向的文件中的空白行数。

分析 :

在案例中学习shell编程

 

解释: 

首先我们知道位置变量有一个是计算参数个数的,即$#,下面我们只要利用好此变量就可以做判断。
为了考虑全面一点,我们需要可以考虑文件是否存在的情况,最后一个难点是如何计算空白的数量。
^$我们知道是空白行,不要忘记考虑里面有空格的情况,所以最后结果为“^[[:space:]]*$”.

答:

#!/bin/bash
#--------------------------
#Filename:argsnumsh
#Revision:3.0
#Date:2017-08-10
#Description:determine args
#--------------------------
[ $# -lt 1 ]&&echo "至少应该给一个参数"&&exit 1
[ -f $1 ]||(echo "please you input exist file";exit 1)||exit 1
Space=`grep "^[[:space:]]*$" $1|wc -l `
[ $# -ge 1 ]&&echo "第一个参数文件的空白行为:${Space}行"&&exit 0
unset Space

4.编写一个脚本/root/bin/createuser.sh,脚本的执行语法必须是:createuser.sh -u username -m password,

选项与参数间可支持多空格,但不能顺序颠倒。当未指定正确的选项或参数时,以错误输出方式提示

“createuser.sh -u username -m password ”后退出脚本。用户名必须以字母开头,可包括数字和_。

否则不合法。以错误输出提示用户"用户名仅包含字母数据和下划线",当用户名检测合法后,判断用户名

是否已存在,若存在,再判断用户是否已设置过密码,若设置过密码,直接退出,未设置,则将密码设置

为所指定的密码后以正确输出方式显示“username 密码已更新后退出”当用户名不存在,则创建用户,

并为该用户设置所指定的密码后以正确输出方式显示“用户username已创建并更新密码”要求脚本执行

过程中不能有非要求的其他输出结果出现。脚本在非正确方式退出时应反回给?参数非0值。

分析:

在案例中学习shell编程

 

解释:

这道题相对上面几道题是有一定难度的,但是细细分析还是有思路可循。首先是确定格式,可以想到会使用

位置变量,$1,$2,$3.. 可以确定的有$1为 -u,$3为-m $2 和$4无法确定,但是可以确定参数个数,即$#为4,

这时,我们使用简单的判断句是可以解决的。下面进行下一个任务,判断用户名是否合法,也就是对$2的判断,

题目中要求用户名必须以字母开头,可包括数字和_,可以使用正则表达式,写出要求,这应该不难即

“^[[:alpha:]]\([[:alnum:]]\|_\)*”这里使用的是标准正则表达式,使用此条件去匹配用户提供的用户名,

不合符提示题目中的语句。接下来判断用户名是否存在,使用的方法当然是使用id命令,若返回值为正确,

那么接下来是判断密码的存在性,使用的方法是看/etc/shadow文件,观察密码位的情况,

若用户不存在时就创建用户,并设置密码,这比较简单,下面解答。

答:

方法一:

#!/bin/bash
#-----------------------
#Filename:createuser.sh
#Revision:7.0
#Date:2017-08-11
#Description:create user
#-----------------------
#判断用户名是否规范,并且给出返回值
decide=`echo $2|grep "^[[:alpha:]]\([[:alnum:]]\|_\)*$" &>/dev/null;echo $?`
#判断用户是否存在,并给出返回值
id=`id $2 &>/dev/null;echo $?`
#截取用户密码位
mima=`getent shadow $2 |cut -d: -f 2`
#判断第一个参数
if [ "$1" != "-u" ];then
         echo "createuser.sh -u username -m password"
         exit 1
fi

#判断第三个参数
if [ "$3" != "-m" ];then
          echo "createuser.sh -u username -m password"
          exit 1
fi
#判断参数个数
if [ $# -ne 4 ];then
          echo " createuser.sh -u username -m password"
          exit 1
fi
#判断第二个参数
if [ $decide -ne 0 ];then
           echo "用户仅包含字母数字和下划线"
           exit 1
else
if [ $id -eq 0 ];then
           if [ "$mima" == "!!" -o "$mima" == "" ];then
           echo "$4" |passwd --stdin $2 &>/dev/null
           echo "$2密码已更新"
           exit 1
else
           exit 1
fi
else

         `useradd $2`
          `echo $4 |passwd --stdin $2 &>/dev/null`
          echo "用户$2已创建并更新密码"

          fi
fi

unset  decide  id mima

方法二:

[ $1 = "-u" -a $3 = "-m" -a $# -eq 4 ] &>/dev/null ||(echo createuser.sh -u username -m password 1>&2 && exit

1)&&(echo$2|egrep "\<^[[:alpha:]]([[:digit:]]|_|[[:alpha:]])*$\>" &>/dev/null ||(echo 用户名仅包含字母数>

据和下划线 1>&2 ;exit 1  ) &&(  `id $2 &>/dev/null` ||(`useradd $2` ;echo $4 |passwd --stdin $2 &>/dev/null &&

echo 用户${2}已创建并更新密码;exit 1

                                ) &&(getent shadow $2 |egrep '\$'[[:digit:]]'\$' &>/dev/null &&(echo 密码已存>

在 1>&2 ;exit 0

                                     )||(

                                echo $4 |passwd --stdin $2 &>/dev/null &&echo 用户${2}密码已更新;exit 1

                                )||exit 1)||exit 1)||exit 1) ||exit 1

总结:方法一更容易理解一点 ,方法二逻辑思维比较强,各有各的好处 。