首页 > 代码库 > Shell第三篇:基本语法

Shell第三篇:基本语法

目录

一、什么是shell script

二、变量

三、运算符

四、流程控制

五、函数

 

  一、什么是shell script

  将OS命令堆积到可执行文件里,由上至下的顺序执行文本里的OS命令,就是脚本了,再加上一些智能(条件/控制)控制,就变成了智能化脚本了

 

二、变量

  part1 为何要有变量

  程序的运行就是一些列状态的变量->用变量值的变化去表示

  part2 变量命名规则

  以字母或下划线开头,剩下的部分可以是:字母、数字、下划线

  最好遵循下述规则:

    1、以字母开头

    2、使用下中中划线或者下划线做单词的链接

    3、同类型的最好用数字区分

    4、对于文件最好加上扩展名

    例如:sql_bak.tar.gz,log_bak.tar.bz2 

  part3 系统变量

    set 和 env 区别

    set:显示所有变量

    env:环境变量

  part4 变量赋值

    VARNAME=VALUE

    echo $VARNAME
    删除变量 unset VARNAME

  part5 常用系统变量

    PATH

    PWD

    LANG

    HOME

    HISTSIZE

    PS1

    IFS

    区域分隔符是空格,换行,TAB健的合集

  part6 全局变量与局部变量

    [root@MiWiFi-R3-srv ~]# gender=‘male‘ #在爹这个位置定义一个局部变量gender

    [root@MiWiFi-R3-srv ~]# export money=1000 #在爹这个位置定义一个全局变量money
    [root@MiWiFi-R3-srv ~]# 
    [root@MiWiFi-R3-srv ~]# 
    [root@MiWiFi-R3-srv ~]# bash #切换到子bash
    [root@MiWiFi-R3-srv ~]# echo $gender #在儿子这里看它爹的局部变量gender,结果为空->看不到

    [root@MiWiFi-R3-srv ~]# echo $money #在儿子这里看它爹的全局变量money,可以看到
    1000
    [root@MiWiFi-R3-srv ~]# 
    [root@MiWiFi-R3-srv ~]# export hobby=‘piao‘ #在儿子这里定义一个全局变量hobby
    [root@MiWiFi-R3-srv ~]# exit #退出,进入爹的bash环境
    exit
    [root@MiWiFi-R3-srv ~]# echo $hobby #爹是看不到儿子的export的,儿子的儿子可以看到

    [root@MiWiFi-R3-srv ~]#

  part7 定义变量名的边界

    [root@MiWiFi-R3-srv ~]# rest_mem=20
    root@MiWiFi-R3-srv ~]# echo ${rest_mem}%
    20% 

   part8 数据类型

    bash中的变量无需声明,直接拿来就能用,默认的变量都会是字符类型,还可以有数字类型,普通的脚本,这两种类型就够用了

 

三、运算符


  part1 算数运算符

    + 、- 、* 、/ 、%

  prat2 关系操作

    与(())连用、< 、> 、<= 、 >= 、== 、!= 、&& 、 ||

  test命令相关,[]可以达到一样的效果

  [root@MiWiFi-R3-srv ~]# x=1

  [root@MiWiFi-R3-srv ~]# [ $x -gt 1 ]
  [root@MiWiFi-R3-srv ~]# echo $?
  0

   part3 赋值运算符

    = 、+= 、*= 、/= 、%=

  [root@MiWiFi-R3-srv ~]# x=10
  [root@MiWiFi-R3-srv ~]# ((x%3))
  [root@MiWiFi-R3-srv ~]# echo $x
  10
  [root@MiWiFi-R3-srv ~]# 
  [root@MiWiFi-R3-srv ~]# ((x%=3))
  [root@MiWiFi-R3-srv ~]# echo $x
  1

  part4 shell里的所有计算器

  $[] (()) $(()) expr bc bc -l

  浮点运算:yum install bc -y

  [root@MiWiFi-R3-srv ~]# echo ‘scale=2;1/3‘|bc -l
  .33

  part5 测试操作

  命令执行后会返回到一个系统变量中$?

  如果$?值为0表示命令执行成功,否则为失败

  

  测试命令text[][[]](())

  打开man test 介绍每一个参数

  1、测试文件状态

  -d 目录

  -s 文件长度 > 0 、非空

  -f 正规文件

  -w 可写

  -r 可读

  -x 可执行

  -L 符号链接

  -u 文件有suid 位置

  2、字符串测试

  = 两个字符串相加

  != 两个字符串不相等

  -z 空串

  -n 非空串

  [root@MiWiFi-R3-srv ~]# var1=‘abc‘
  [root@MiWiFi-R3-srv ~]# var2=‘123‘
  [root@MiWiFi-R3-srv ~]# [ $var1 == $var2 ]
  [root@MiWiFi-R3-srv ~]# echo $?
  1

  3、测试数值

  -ep 等于

  -ne 不等于

  -gt 大于

  -lt 小于

  -ge 大于等于

  -le 小于等于

  [root@MiWiFi-R3-srv ~]# [ 10000 -gt 250 ] #不要使用大于号小于号等于号等,要使用man test中规定的,详见下一小节4拓展
  [root@MiWiFi-R3-srv ~]# echo $?
  0

  4、拓展测试符号

  数字测试符号

  # [ 10 < 2 ] # 语法错误
  -bash: 2: 没有那个文件或目录
  # 

  # [[ 2 > 10 ]] # 结果错误
  # echo $?
  0
  # [[ 20 > 10 ]] # 正确
  # echo $?
  0
  # (( 10 < 20 ))
  # echo $?
  0

  字符测试
  # [ "aa" = "aa" ]
  # echo $?
  0
  # [[ "aa" = "aa" ]]
  # echo $?
  0
  # (( "aa" = "aa" )) #结果错误
  # echo $?
  1
  混合测试
  # [ a = a -a 10 < 20 ]
  -bash: 20: 没有那个文件或目录
  [root@seker ~]# [[ a = a -a 10 < 20 ]]
  -bash: syntax error in conditional expression
  -bash: syntax error near `-a‘
  [root@seker ~]# [[ a = a && 10 < 20 ]]
  [root@seker ~]# echo $?
  0
  [root@seker ~]# [[ a = a || 10 < 20 ]]
  [root@seker ~]# echo $?
  0
  [root@seker ~]# (( a = a || 10 < 20 ))
  [root@seker ~]# echo $?
  0
  [root@seker ~]# (( a = a && 10 < 20 ))
  [root@seker ~]# echo $?
  0
  [root@seker ~]# 
  

四、流程控制

  part1 分支结构

  

#!/bin/bash
var=‘/etc/init.d‘
#var=‘/dev/sda‘
if [ -d $var ]
    then
        echo "$var is directory"
elif [ -b $var ]
    then
        echo "$var is block"
elif [ -f $var ]
    then
        echo "$var is regular file"
else
        echo ‘unknow‘
fi

   if 测试中还可以执行命令 根据命令的返回值做判断
  # if cd / ;then echo Y ;fi
  # if grep -q root /etc/passwd ;then echo Y ;fi

  

#!/bin/bash
username=egon
password=123
read -p user:  name 
read -p passwd:  passwd

if [ $name = $username -a $passwd = $password ];then
    echo login successful
else
    echo username or password err
fi

  

#!/bin/bash
age=87
read -p num:  n

if [ $n -eq $age ];then
    echo you get it
elif [ $n -gt $age ];then
    echo too big
elif [ $n -lt $age ];then
    echo too small
fi

  

#!/bin/bash
read -p your score:  score

if [ $score -ge 90  ];then
    echo 优秀
elif [ $score -ge 70 -a $score -lt 90 ];then
    echo 良好
elif [ $score -ge 60 -a $score -lt 70 ];then
    echo 一般
elif [ $score -lt 60 ];then
    echo 较差
fi

 

   向脚本传递参数

  

#test.sh
echo $0
echo $1
echo $2
echo $3
echo ${11}
echo ‘$$‘ $$
echo ‘$*‘ $*
echo ‘$@‘ $@
echo ‘$#‘ $#
echo ‘$?‘ $?

‘‘‘
测试:python test.sh 1 2 3 4 5 6 7 8 9 10 11
输出结果:
./test.sh
2
11
$$ 14312
$* 1 2 3 4 5 6 7 8 9 10 11
$@ 1 2 3 4 5 6 7 8 9 10 11
$# 11
$? 0
‘‘‘

   修改脚本,使其能接收调用者传来的参数

  

[root@MiWiFi-R3-srv ~]# cat test_file.sh 
#!/bin/bash
if [ -d $1 ]
    then
        echo "$1 is directory"
elif [ -b $1 ]
    then
        echo "$1 is block"
elif [ -f $1 ]
    then
        echo "$1 is regular file"
else
        echo ‘unknown‘
fi
[root@MiWiFi-R3-srv ~]# ./test_file.sh /etc/passwd
/etc/passwd is regular file

   part2 循环结构

  while 循环

  while (条件)

  do 

  动作

  done

  需要无限循环时会选择while

  

[root@MiWiFi-R3-srv ~]# cat login.sh 
#!/bin/bashwhile :
do
    read -p ‘please input your name: ‘ name
    read -p ‘please input your password: ‘ pwd
    if [ $name = ‘egon‘ ] && [ $pwd = ‘123‘ ]
        then
            echo ‘login sucessful‘
            break #continue
    fi
done
[root@MiWiFi-R3-srv ~]# ./login.sh 
please input your name: egon
please input your

 

[root@MiWiFi-R3-srv ~]# cat 1.sh 
#!/bin/bash
i=1
while ((i<10))
do
    echo $i
    ((i++))
done
[root@MiWiFi-R3-srv ~]# ./1.sh 
2
4
6
8

   while 死循环

  

[root@MiWiFi-R3-srv ~]# cat 1.sh 
#!/bin/bash

var1=AAA
var2=BBB
var3=CCC
while :
do
    clear
    echo -e "A:${var1}\nB:${var2}\nC:${var3}"
    temp=$var1
    var1=$var2
    var2=$var3
    var3=$temp
    sleep 1
done

   wihle和read实现逐行处理

  

[root@MiWiFi-R3-srv ~]# cat 1.sh 
#!/bin/bash
while read var
do
    echo $((++i)):$var
done</etc/passwd


[root@MiWiFi-R3-srv ~]# ./1.sh 
1:root:x:0:0:root:/root:/bin/bash
2:bin:x:1:1:bin:/bin:/sbin/nologin
3:daemon:x:2:2:daemon:/sbin:/sbin/nologin
4:adm:x:3:4:adm:/var/adm:/sbin/nologin
...........

   for 循环

  shell 格式的for循环

  

for i in {1..10}
do
echo $i
done

   shell的for,常用in列表方式

  

for i in 1 2 3
for i in {1,2,3}
for i in {1..9}
for i in {9..1}
for i in {a..z}
for i in {A..Z}
for i in {X..Z}
for i in $(cmd)
for i in $(find ...)

   例子:

  

检查内网存活的IP
    #!/bin/bash

    for i in {1..254}
    do
            (ping -W 1 -c 1 192.168.1.$i &> /dev/null && echo 192.168.1.$i) &
    done
    
    
    让文件测试脚本支持多个参数
    #!/bin/bash
    
    for i in $@
    do
      if [[ -d $i ]]
      then
        echo "$i is directory."
      elif [[ -b $i  ]]
      then
        echo "$i is block device."
      elif [[ -f $i ]]
      then
        echo "$i is a regular file."
      else
        echo "unknow."
      fi
    done

   多个for嵌套

  嵌套for 中使用

  continue:默认退出本次循环

  break:默认退出本层循环

  

使用break:
break 默认参数是 1 
所以写 break 等于 break 1
意义:退出当前循环层
break 2 则向上退出2层循环 当前循环也计算在退出层次里

for i in {1..9}
do
for j in {0..9} 
do
for n in {0..9}
do
echo $i$j$n
if ((n==5))
then
break 3
fi
done
done
done


使用continue
continue = continue 1
在当次循环中忽略continue后续的代码
就是:立即结束当前循环中的当次循环,而转入当前循环的下一次循环

continue 2 = break 1
continue 3 = break 2
....
依次类推

for i in {1..9}
do
for j in {0..9}
do
for n in {0..9}
do
echo $i$j$n
if ((n==5))
then
continue
echo "-----------------------------"
fi
done
done
done

   

  可以直接在命令行写for循环

  # for i in {1..10};do [ $i -eq 5 ] && continue || echo $i;done
  # for i in {1..10};do [ $i -eq 5 ] && break || echo $i;done

  

  练习:
  统计/dev下每种类型文件的数量

  

#!/bin/bash
dir=‘/dev‘
for i in `ls $dir`
do
    if [ -b $dir/$i ]
    then
        ((block++))
    elif [ -f $dir/$i ]
    then
        ((file++))
         
    elif [ -d $dir/$i ]
    then
        ((directory++))
    else
        ((unkown++))
    fi
done

echo ‘block‘ $block
echo ‘regular file‘ $file
echo ‘directory‘ $directory
echo ‘unkown‘ $unkown

   向脚本传递一个用户名,验证这个用户是否存在

  

[root@MiWiFi-R3-srv ~]# cat testuser.sh 
#!/bin/bash
id $1 &> /dev/null
if [ $? -eq 0 ];then
    echo "用户$1存在"
else
    echo "用户$1不存在"
fi
[root@MiWiFi-R3-srv ~]# ./testuser.sh root
用户root存在

   添加30个用户,再将它们删除

  

for i in {1..30};
do
    useradd user$i&&echo "user$i create successful"
done


for i in {1..30};
do
    userdel -r user$i&&echo "user$i delete successful"
done

   part3 case语句

  

  read -p "username: " -t 5 uname
  echo
  if [[ -z $uname ]]
  then
  uname=default
  fi

  case $uname in
  root)
  echo "welcome $uname"
  ;;
  seker)
  echo "welcome $uname"
  ;;
  default)
  echo "welcome $uname"
  ;;
  *)
  echo "no user $uname"
  esac

  part4 结合一起,制作一个简单的菜单功能

  

#!/bin/bash
echo "script name: `basename $0`"
echo "version 1.0"
echo "date 2017-03-23"
echo "Author: biubiu"
while read -p "(h for help): " var
do
    case $var in 
        p|P|cpu|CPU)
            echo -e "\n\n"
            grep ‘model name\|cpu MHz\|processor‘ /proc/cpuinfo |sort |uniq
            echo -e "\n\n"
        ;;
        m|m|mem|MEM)
            echo -e "\n\n"
            free
            echo -e "\n\n"
        ;;
        d|D|disk|DISK)
            echo -e "\n\n"
            df -Th
            echo -e "\n\n"
        ;;
        h|H|help|HELP)
            echo -e "\n\tcommand\taction\n\n"
            for i in cpu mem disk
            do
            echo -e "\t$i\t${i}_info"
            done
            echo -e "\thelp\tthis help page.."
            echo -e "\tquit\texit !!.."
            echo -e "\n\n"
        ;;
        q|Q|quit|exit)
            exit
        ;;
        *)
            echo -e "\n$var Enter Error...\n"
    esac
done

   附加:

  

# cat select.sh 
#!/bin/bash

PS3=‘choose one: ‘ #select 默认使用PS3做提示符
echo
select var in $(for i in {A..D};do echo $i;done)
do
echo
echo "your choose is $var"
echo "OK"
echo
break # 跳出select,否则是死循环
done
# 
# ./select.sh

1) A
2) B
3) C
4) D
choose one: 3

your choose is C
OK

#
若省略 in list 则把 $@ 做列表项
# cat select.sh 
#!/bin/bash

PS3=‘choose one: ‘
echo
select var 
do
echo
echo "your choose is $var"
echo "OK"
echo
break
done
#
# ./select.sh A B C D

1) A
2) B
3) C
4) D
choose one: C

your choose is 
OK

了解:select 菜单

 

五、函数

  交互shell中的函数

  function abc(){echo‘aaa‘;echo‘bbbb‘;}

  set 查看

  调用 abc

  

  [root@seker ~]# abc
  aaa
  bbbb
  [root@seker ~]# 


  脚本中的函数

  1.函数定义
  shell允许将一组命令集或语句形成一个可用块,这些块称为shell函数
  定义函数的格式:

  function-name (){
  command1
  ........
  }
  或 function function-name(){ #函数名前面多了个function关键字
  command1
  ........
  }
  2.函数调用
  以下是一个函数的脚本实例:
  #!/bin/bash
  #hello
  function hello(){ #声明函数
  echo "Hello!" #函数的主体,输出"Hello!"
  } #函数结束
  hello #调用函数
  3.参数传递
  向函数传递参数就像在脚本是使用变量位置$1,$2,$3...$9
  以下是一个传递参数的实例:
  #!/bin/bash
  #hellofun
  function hello(){
  echo "Hello! The first parameter is ‘$1‘."
  }
  hello good
  #该脚本执行的结果是: Hello! The first parameter is ‘good‘.
  4.函数文件
  保存函数的文件,用以上的例子写成一个函数文件如下:
  #!/bin/bash
  #hellofunction
  function hello(){
  echo "Hello!"
  return 1
  }
  上面的hellofunction文件就是一个函数文件,可通过另一个脚本来调用
  #!/bin/bash
  #hellof
  . hellofunction #注意点和hellofunction之间有个空格
  hello
  5.载入和删除
  用set查看已载入的函数
  用unset function-name 取消载入
  举例如下:
  #!/bin/bash
  #hellof
  . hellofunction
  unset hello
  hello #因为已经取消载入。。所以会出错
  6.函数返回状态
  #!/bin/bash
  #hellofun
  function hello(){
  echo "Hello! The first parameter is ‘$1‘."
  return 1
  }
  hello
  echo $? #输出返回的状态值(一般成功是返回0,其它值为失败)

 

 

   

 

Shell第三篇:基本语法