一文吃透shell编写工具及基本法则!

电子说

1.3w人已加入

描述

前言

大家好,这里是浩道linux,主要给大家分享linux、python、网络通信相关的IT知识平台。

今天浩道跟大家分享关于shell编程相关的硬核干货,通过本文你将吃透shell编写相关工具及基本法则。让你编写shell脚本如鱼得水!

 文章来源:资源整理综合自网络

1. 文件处理工具

1.1 grep工具

过滤

 

grep用于根据关键字进行行过滤
grep options 'keys' filename
OPTIONS:
    -i: 不区分大小写
    -v: 查找不包含指定内容的行,反向选择
    -w: 按单词搜索
    -o: 打印匹配关键字
    -c: 统计匹配到的次数
    -n: 显示行号
    -r: 逐层遍历目录查找
    -A: 显示匹配行及后面多少行    
    -B: 显示匹配行及前面多少行
    -C: 显示匹配行前后多少行
    -l:只列出匹配的文件名
    -L:列出不匹配的文件名
    -e: 使用正则匹配
    -E:使用扩展正则匹配
    ^key:以关键字开头
    key$:以关键字结尾
    ^$:匹配空行
    --color=auto :可以将找到的关键词部分加上颜色的显示

临时设置:
# alias grep='grep --color=auto'            //只针对当前终端和当前用户生效

永久设置:
1)全局(针对所有用户生效)
vim /etc/bashrc
alias grep='grep --color=auto'
source /etc/bashrc

2)局部(针对具体的某个用户)
vim ~/.bashrc
alias grep='grep --color=auto'
source ~/.bashrc

示例:
# grep -i root passwd                忽略大小写匹配包含root的行
# grep -w ftp passwd                 精确匹配ftp单词
# grep -w hello passwd                 精确匹配hello单词;自己添加包含hello的行到文件中
# grep -wo ftp passwd                 打印匹配到的关键字ftp
# grep -n root passwd                 打印匹配到root关键字的行好
# grep -ni root passwd                 忽略大小写匹配统计包含关键字root的行
# grep -nic root passwd                忽略大小写匹配统计包含关键字root的行数
# grep -i ^root passwd                 忽略大小写匹配以root开头的行
# grep bash$ passwd                             匹配以bash结尾的行
# grep -n ^$ passwd                             匹配空行并打印行号
# grep ^# /etc/vsftpd/vsftpd.conf        匹配以#号开头的行
# grep -v ^# /etc/vsftpd/vsftpd.conf    匹配不以#号开头的行
# grep -A 5 mail passwd                     匹配包含mail关键字及其后5行
# grep -B 5 mail passwd                     匹配包含mail关键字及其前5行
# grep -C 5 mail passwd                     匹配包含mail关键字及其前后5行

 

1.2 cut工具

截取

 

cut用于列截取
-c:    以字符为单位进行分割。
-d:    自定义分隔符,默认为制表符。	
-f:    与-d一起使用,指定显示哪个区域。

# cut -d: -f1 1.txt             以:冒号分割,截取第1列内容
# cut -d: -f1,6,7 1.txt     以:冒号分割,截取第1,6,7列内容
# cut -c4 1.txt                 截取文件中每行第4个字符
# cut -c1-4 1.txt             截取文件中每行的1-4个字符
# cut -c4-10 1.txt             
# cut -c5- 1.txt                 从第5个字符开始截取后面所有字符

课堂练习:
用小工具列出你当系统的运行级别。5/3

 

1.3 sort工具

排序

 

sort:将文件的每一行作为一个单位,从首字符向后,依次按ASCII码值进行比较,最后将他们按升序输出。

-u :去除重复行
-r :降序排列,默认是升序
-o : 将排序结果输出到文件中  类似 重定向符号>
-n :以数字排序,默认是按字符排序
-t :分隔符
-k :第N列
-b :忽略前导空格。
-R :随机排序,每次运行的结果均不同。

 示例:
# sort -n -t: -k3 1.txt             按照用户的uid进行升序排列
# sort -nr -t: -k3 1.txt             按照用户的uid进行降序排列
# sort -n 2.txt                         按照数字排序
# sort -nu 2.txt                         按照数字排序并且去重
# sort -nr 2.txt 
# sort -nru 2.txt 
# sort -nru 2.txt 
# sort -n 2.txt -o 3.txt         按照数字排序并将结果重定向到文件
# sort -R 2.txt 
# sort -u 2.txt 

 

1.4 uniq工具

去除连续重复

 

uniq:去除连续重复行
-i: 忽略大小写
-c: 统计重复行次数
-d:只显示重复行

# uniq 2.txt 
# uniq -d 2.txt 
# uniq -dc 2.txt 

 

1.5 tee工具

 

tee工具从标准输入读取并写入标准输出和文件,即:双向覆盖重定向<屏幕输出|文本输入>
-a 双向追加重定向

# echo hello world
# echo hello world|tee file1
# cat file1 
# echo 999|tee -a file1
# cat file1 

 

1.6 paste工具

 

paste工具用于合并文件行

-d:自定义间隔符,默认是tab
-s:串行处理,非并行

[root@server shell01]# cat a.txt 
hello
[root@server shell01]# cat b.txt 
hello world
888
999
[root@server shell01]# paste a.txt b.txt 
hello   hello world
        888
        999
[root@server shell01]# paste b.txt a.txt   
hello world     hello
888
999

[root@server shell01]# paste -d'@' b.txt a.txt 
hello world@hello
888@
999@

[root@server shell01]# paste -s b.txt a.txt 
hello world     888     999
hello

 

1.7 tr工具

==字符转换:替换,删除==

 

tr用来从标准输入中通过替换或删除操作进行字符转换;主要用于删除文件中控制字符或进行字符转换。
使用tr时要转换两个字符串:字符串1用于查询,字符串2用于处理各种转换。

语法:
commands|tr  'string1'  'string2'
tr  'string1'  'string2' < filename

tr options 'string1' < filename

-d 删除字符串1中所有输入字符。
-s 删除所有重复出现字符序列,只保留第一个;即将重复出现字符串压缩为一个字符串。


    a-z 任意小写
    A-Z 任意大写
    0-9 任意数字
  []       all letters and digits        所有字母和数字
  []       all letters                        所有字母
  []       all horizontal whitespace    所有水平空白
  []       all control characters        所有控制字符
 Ctrl-H          退格符
f Ctrl-L          走行换页

 Ctrl-J          新行

 Ctrl-M          回车
	 Ctrl-I          tab键
  []    all digits    所有数字
  []    all printable characters, not including space
  所有可打印的字符,不包含空格
  []       all lower case letters        所有小写字母
  []       all printable characters, including space
  所有可打印的字符,包含空格
  []       all punctuation characters            所有的标点符号
  []       all horizontal or vertical whitespace    所有水平或垂直的空格
  []       all upper case letters                所有大写字母
  []      all hexadecimal digits                所有十六进制数字
  [=CHAR=]        all characters which are equivalent to CHAR    所有字符



[root@server shell01]# cat 3.txt     自己创建该文件用于测试
ROOT0root:/root:/bin/bash
bin1bin:/bin:/sbin/nologin
daemon2daemon:/sbin:/sbin/nologin
adm3adm:/var/adm:/sbin/nologin
lp4lp:/var/spool/lpd:/sbin/nologin
sync5sync:/sbin:/bin/sync
shutdown6shutdown:/sbin:/sbin/shutdown
halt7halt:/sbin:/sbin/halt
mail8mail:/var/spool/mail:/sbin/nologin
uucp10uucp:/var/spool/uucp:/sbin/nologin
boss02516:/home/boss02:/bin/bash
vip517:/home/vip:/bin/bash
stu1518:/home/stu1:/bin/bash
mailnull47:/var/spool/mqueue:/sbin/nologin
smmsp51:/var/spool/mqueue:/sbin/nologin
aaaaaaaaaaaaaaaaaaaa
bbbbbb111111122222222222233333333cccccccc
hello world 888
666
777
999


# tr -d '[:/]' < 3.txt                 删除文件中的:和/
# cat 3.txt |tr -d '[:/]'            删除文件中的:和/
# tr '[0-9]' '@' < 3.txt             将文件中的数字替换为@符号
# tr '[a-z]' '[A-Z]' < 3.txt         将文件中的小写字母替换成大写字母
# tr -s '[a-z]' < 3.txt             匹配小写字母并将重复的压缩为一个
# tr -s '[a-z0-9]' < 3.txt         匹配小写字母和数字并将重复的压缩为一个
# tr -d '[]' < 3.txt         删除文件中的数字
# tr -d '[]' < 3.txt         删除水平空白
# tr -d '[]' < 3.txt         删除所有水平和垂直空白

 

小试牛刀

使用小工具分别截取当前主机IP;截取NETMASK;截取广播地址;截取MAC地址

 

[root@server shell01]# ifconfig eth0|grep 'Bcast'|tr -d '[a-zA-Z ]'|cut -d: -f2,3,4
10.1.1.1255.255.255.0
[root@server shell01]# ifconfig eth0|grep 'Bcast'|tr -d '[a-zA-Z ]'|cut -d: -f2,3,4|tr ':' '
'
10.1.1.1
10.1.1.255
255.255.255.0
[root@server shell01]# ifconfig eth0|grep 'HWaddr'|cut -d: -f2-|cut -d' ' -f4
0029AE:54

# ifconfig eth1|grep Bcast|cut -d: -f2|cut -d' ' -f1
# ifconfig eth1|grep Bcast|cut -d: -f2|tr -d '[ a-zA-Z]'
# ifconfig eth1|grep Bcast|tr -d '[:a-zA-Z]'|tr ' ' '@'|tr -s '@'|tr '@' '
'|grep -v ^$
# ifconfig eth0|grep 'Bcast'|tr -d []|tr '[ :]' '
'|grep -v ^$
# ifconfig eth1|grep HWaddr|cut -d ' ' -f11
# ifconfig eth0|grep HWaddr|tr -s ' '|cut -d' ' -f5
# ifconfig eth1|grep HWaddr|tr -s ' '|cut -d' ' -f5

 

将系统中所有普通用户的用户名、密码和默认shell保存到一个文件中,要求用户名密码和默认shell之间用tab键分割

 

[root@server shell01]# grep 'bash$' passwd |grep -v '^root'|cut -d: -f1,2,7|tr ':' '	'
stu1    x       /bin/bash
code    x       /bin/bash
kefu    x       /bin/bash
kefu1   x       /bin/bash
kefu2   x       /bin/bash
user01  x       /bin/bash
stu2    x       /bin/bash
[root@server shell01]# grep bash$ passwd |grep -viE 'root|mysql'|cut -d: -f1,2,7|tr ':' '	' |tee a.txt

注释:
-E 匹配扩展正则表达式,|代表或者,是一个扩展正则

 

2. 编程语言分类

编译型语言:

==程序在执行之前需要一个专门的编译过程==,把程序编译成为机器语言文件,运行时不需要重新翻译,直接使用编译的结果就行了。程序执行效率高,依赖编译器,跨平台性差些。如C、C++

解释型语言:

程序不需要编译,程序在运行时由==解释器==翻译成机器语言,每执行一次都要翻译一次。因此效率比较低。比如Python/JavaScript/ Perl /ruby/Shell等都是解释型语言。

脚本

/语言分类

总结:

编译型语言比解释型语言==速度较快==,但是不如解释型语言==跨平台性好==。如果做底层开发或者大型应用程序或者操作系开发一==般都用编译型语言==;如果是一些服务器脚本及一些辅助的接口,对速度要求不高、对各个平台的==兼容性有要求==的话则一般都用==解释型语言==。

3. shell介绍

脚本

00_shell介绍

总结:

==shell就是人机交互的一个桥梁==

shell的种类

 

[root@MissHou ~]# cat /etc/shells 
/bin/sh            #是bash shell的一个快捷方式
/bin/bash        #bash shell是大多数Linux默认的shell,包含的功能几乎可以涵盖shell所有的功能
/sbin/nologin    #表示非交互,不能登录操作系统
/bin/dash        #小巧,高效,功能相比少一些
/bin/tcsh        #是csh的增强版,完全兼容csh
/bin/csh        #具有C语言风格的一种shell,具有许多特性,但也有一些缺陷

 

用户在终端(终端就是bash的接口)输入命令

|
               bash          //bash就是shell的一种类型(bash shell)
        |
       kernel
        |  
       物理硬件等

4. shell脚本

什么是shell脚本?

一句话概括

简单来说就是将需要执行的命令保存到文本中,==按照顺序执行==。它是解释型的,意味着不需要编译。

准确叙述

若干命令 + 脚本的基本格式 + 脚本特定语法 + 思想= shell脚本

什么时候用到脚本?

重复化、复杂化的工作,通过把工作的命令写成脚本,以后仅仅需要执行脚本就能完成这些工作。

①自动化分析处理

②自动化备份

③自动化批量部署安装

④等等…

如何学习shell脚本?

尽可能记忆更多的命令

掌握脚本的标准的格式(指定魔法字节、使用标准的执行方式运行脚本)

必须==熟悉掌握==脚本的基本语法(重点)

脚本的基本写法:

 

#!/bin/bash
//脚本第一行, #!魔法字符,指定脚本代码执行的程序。即它告诉系统这个脚本需要什么解释器来执行,也就是使用哪一种Shell

//以下内容是对脚本的基本信息的描述
# Name: 名字
# Desc:描述describe
# Path:存放路径
# Usage:用法
# Update:更新时间

//下面就是脚本的具体内容
commands
...

 

脚本执行方法:

标准脚本执行方法(建议):(魔法字节指定的程序会生效)

 

[root@MissHou shell01]# cat 1.sh 
#!/bin/bash
#xxxx
#xxx
#xxx
hostname
date
[root@MissHou shell01]# chmod +x 1.sh 
[root@MissHou shell01]# ll
total 4
-rwxr-xr-x 1 root root 42 Jul 22 14:40 1.sh
[root@MissHou shell01]# /shell/shell01/1.sh 
MissHou.itcast.cc
Sun Jul 22 1400 CST 2018
[root@MissHou shell01]# ./1.sh 
MissHou.itcast.cc
Sun Jul 22 1430 CST 2018

 

非标准的执行方法(不建议):(魔法字节指定的程序不会运作)

~~~shell
[root@MissHou shell01]# bash 1.sh
MissHou.itcast.cc
Sun Jul 22 1451 CST 2018
[root@MissHou shell01]# sh 1.sh
MissHou.itcast.cc
Sun Jul 22 1401 CST 2018
[root@MissHou shell01]#
[root@MissHou shell01]# bash -x 1.sh

hostname
MissHou.itcast.cc

date
Sun Jul 22 1420 CST 2018

-x:一般用于排错,查看脚本的执行过程
-n:用来查看脚本的语法是否有问题

注意:如果脚本没有加可执行权限,不能使用标准的执行方法执行,bash 1.sh

其他:
[root@server shell01]# source 2.sh
server
Thu Nov 22 1550 CST 2018
[root@server shell01]# . 2.sh
server
Thu Nov 22 1507 CST 2018

source 和 . 表示读取文件,执行文件里的命令
~~~

5. bash基本特性

5.1 命令和文件自动补全

Tab只能补全命令和文件 (RHEL6/Centos6)

5.2 常见的快捷键

 

^c               终止前台运行的程序
^z                将前台运行的程序挂起到后台
^d               退出 等价exit
^l               清屏 
^a |home      光标移到命令行的最前端
^e |end      光标移到命令行的后端
^u               删除光标前所有字符
^k               删除光标后所有字符
^r                搜索历史命令

 

5.3 常用的通配符(重点)

 

*:    匹配0或多个任意字符
?:    匹配任意单个字符
[list]:    匹配[list]中的任意单个字符
[!list]: 匹配除list中的任意单个字符
{string1,string2,...}:匹配string1,string2或更多字符串

举例:
touch file{1..3}
touch file{1..13}.jpg
# ls file*
# ls *.jpg
# ll file?
# ll file?.jpg
# ll file??.jpg
# ll file1?.jpg
# ll file?.jpg
# ll file[1023].jpg
# ll file[0-13].jpg
# ll file1[0-9].jpg
# ll file[0-9].jpg
# ll file?[1-13].jpg
# ll file[1,2,3,10,11,12].jpg
# ll file1{11,10,1,2,3}.jpg
# ll file{1..10}.jpg
# ll file{1...10}.jpg

 

5.4 bash中的引号(重点)

双引号""   :会把引号的内容当成整体来看待,允许通过符号引用其他变量值

  • 单引号′′:会把引号的内容当成整体来看待,禁止引用其他变量值,shell中特殊符号都被视为普通字符
  • 反撇号‘‘:反撇号和()一样,引号或括号里的命令会优先执行,如果存在嵌套,反撇号不能用

     

    [root@server dir1]# echo "$(hostname)"
    server
    [root@server dir1]# echo '$(hostname)'
    $(hostname)
    [root@server dir1]# echo "hello world"
    hello world
    [root@server dir1]# echo 'hello world'
    hello world
    
    [root@server dir1]# echo $(date +%F)
    2018-11-22
    [root@server dir1]# echo `echo $(date +%F)`
    2018-11-22
    [root@server dir1]# echo `date +%F`
    2018-11-22
    [root@server dir1]# echo `echo `date +%F``
    date +%F
    [root@server dir1]# echo $(echo `date +%F`)
    2018-11-22
    

     

    变量的定义

    1. 变量的分类

    本地变量:当前用户自定义的变量。当前进程中有效,其他进程及当前进程的子进程无效。

    环境变量:当前进程有效,并且能够被子进程调用。

    查看当前用户的环境变量                        env

    查询当前用户的所有变量(临时变量与环境变量)         **set **

    **export    //将当前变量变成环境变量 **

     

    [root@MissHou tmp]# export A=hello        //临时将一个本地变量(临时变量)变成环境变量
    [root@MissHou tmp]# env|grep ^A
    A=hello
    
    永久生效:
    vim /etc/profile 或者 ~/.bashrc
    export A=hello
    或者
    A=hello
    export A
    
    说明:系统中有一个变量PATH,环境变量
    export PATH=/usr/local/mysql/bin:$PATH
    

     

    全局变量:全局所有的用户和程序都能调用,且继承,新建的用户也默认能调用.

     

    $HOME/.bashrc             当前用户的bash信息(aliase、umask等)
    $HOME/.bash_profile      当前用户的环境变量()
    oracle——>oracle用户——>$oracle/.bash_profile——>export home_install=/u01/app/xxx
    
    $HOME/.bash_logout          每个用户退出当前shell时最后读取的文件
    
    /etc/bashrc             使用bash shell用户全局变量
    grep --color=auto
    umask
    
    /etc/profile              系统和每个用户的环境变量信息
    
    mycat_home=/usr/local/mycat/bin
    export mycat_home
    执行mycat命令
    # mycat
    $ mycat
    
    用户登录系统读取相关文件的顺序:
    /etc/profile——>$HOME/.bash_profile——>$HOME/.bashrc——>/etc/bashrc——>$HOME/.bash_logout
    
    source /etc/bashrc
    

     

    系统变量(内置bash中变量) :shell本身已经固定好了它的名字和作用.

     

    $?:上一条命令执行后返回的状态,当返回状态值为0时表示执行正常,非0值表示执行异常或出错
     若退出状态值为0,表示命令运行成功
     若退出状态值为127,表示command not found
     若退出状态值为126,表示找到了该命令但无法执行(权限不够)
     若退出状态值为1&2,表示没有那个文件或目录
    
    $$:当前所在进程的进程号     echo $$   eg:kill -9 `echo $$`  = exit   退出当前会话
    $!:后台运行的最后一个进程号  (当前终端)  # gedit &
    !$    调用最后一条命令历史中的参数
    !!    调用最后一条命令历史
    
    
    $#:脚本后面接的参数的个数
    $*:脚本后面所有参数,参数当成一个整体输出,每一个变量参数之间以空格隔开
    $@: 脚本后面所有参数,参数是独立的,也是全部输出
    
    $0:当前执行的进程/程序名  echo $0        
    $1~$9 位置参数变量
    ${10}~${n} 扩展位置参数变量  第10个位置变量必须用{}大括号括起来
    ./1.sh a b c
    
    [root@MissHou shell01]# cat 2.sh 
    #!/bin/bash
    #xxxx
    echo "$0 = $0"
    echo "$# = $#"
    echo "$* = $*"
    echo "$@ = $@"
    echo "$1 = $1" 
    echo "$2 = $2" 
    echo "$3 = $3" 
    echo "$11 = ${11}" 
    echo "$12 = ${12}" 
    
    了解$*和$@的区别:
    $*    :表示将变量看成一个整体
    $@    :表示变量是独立的
    
    #!/bin/bash
    for i in "$@"
    do
    echo $i
    done
    
    echo "======我是分割线======="
    
    for i in "$*"
    do
    echo $i
    done
    
    [root@MissHou shell01]# bash 3.sh a b c
    a
    b
    c
    ======我是分割线=======
    a b c
    

     

    2. 什么时候需要定义变量?

    如果某个内容需要多次使用,并且在代码中重复出现,那么可以用变量代表该内容。这样在修改内容的时候,仅仅需要修改变量的值。

    在代码运作的过程中,可能会把某些命令的执行结果保存起来,后续代码需要使用这些结果,就可以直接使用这个变量。

    3. 变量的定义规则

     

    1. 默认情况下,shell里定义的变量是不分类型的,可以给变量赋与任何类型的值;等号两边不能有空格,对于有空格的字符串做为赋值时,要用引号引起来
    变量名=变量值
    
    2. 变量的获取方式:    $变量名     ${变量名}   
    [root@MissHou shell01]# a=12345678
    [root@MissHou shell01]# echo $a
    12345678
    [root@MissHou shell01]# echo ${a}
    12345678
    [root@MissHou shell01]# echo ${a3}        a表示变量名;2表示从第3个字符开始;3表示后面3个字符
    345
    
    如果获取变量的全部两个都可以;如果获取变量的某一部分,用${}
    
    3. 取消变量:     unset  变量名
    
    4. 变量名区分大小写,同名称但大小写不同的变量名是不同的变量
    5. 变量名可以是字母或数字或下划线,但是不能以数字开头或者特殊字符
    [root@MissHou shell01]# 1a=hello
    -bash: 1a=hello: command not found
    [root@MissHou shell01]# ?a=hello
    -bash: ?a=hello: command not found
    [root@MissHou shell01]# _a=hello
    [root@MissHou shell01]# echo $_a
    hello
    
    6. 命令的执行结果可以保存到变量
    [root@server shell01]# kernel=`uname -r`
    [root@server shell01]# echo $kernel
    2.6.32-431.el6.x86_64
    [root@server shell01]# name=$(uname -n)
    [root@server shell01]# echo $name
    server.itcast.cc
    
    7. 有类型变量 declare
    -i 将变量看成整数 
    -r 使变量只读  readonly
    -x 标记变量通过环境导出  export
    -a    指定为索引数组(普通数组);查看普通数组
    -A 指定为关联数组;查看关联数组
    
    [root@server shell01]# a=10
    [root@server shell01]# b=20
    [root@server shell01]# echo $a+$b
    10+20
    
    [root@server shell01]# declare -i a=2
    [root@server shell01]# declare -i b=4
    [root@server shell01]# declare -i c=$a+$b
    [root@server shell01]# echo $c
    6
    
    [root@server shell01]# AAAA=hello
    [root@server shell01]# export AAAA
    [root@server shell01]# env|grep AAAA
    AAAA=hello
    [root@server shell01]# declare -x BBBB=hello
    [root@server shell01]# env|grep BBBB
    BBBB=hello
    
    
    8. 数组
    普通数组:只能使用整数作为数组索引(元素的下标)
    关联数组:可以使用字符串作为数组索引(元素的下标)
    
    普通数组定义:用括号来表示数组,数组元素(变量)用“空格”符号分割开。定义数组的一般形式为:
    一次赋一个值:
    变量名=变量值
    array[0]=v1
    array[1]=v2
    array[3]=v3
    一次赋多个值:
    array=(var1 var2 var3 var4)
    array1=(`cat /etc/passwd`)            //将文件中每一行赋值给array1数组
    array2=(`ls /root`)
    array3=(harry amy jack "Miss Hou")
    array4=(1 2 3 4 "hello world" [10]=linux)
    
    读取数组:
    ${array[i]}  i表示元素的下标
    使用@ 或 * 可以获取数组中的所有元素:
    获取第一个元素
    echo ${array[0]}
    echo ${array[*]}            获取数组里的所有元素
    echo ${#array[*]}            获取数组里所有元素个数
    echo ${!array[@]}        获取数组元素的索引下标
    echo ${array[@]2}    访问指定的元素;1代表从下标为1的元素开始获取;2代表获取后面几个元素
    
    
    [root@server shell01]# array[0]=var1
    [root@server shell01]# array[1]=var2
    [root@server shell01]# array[2]=var3
    [root@server shell01]# array1=(uu1 uu2 uu3 uu4)
    [root@server shell01]# ls
    1.sh  2.sh  3.sh  4.sh  passwd
    [root@server shell01]# array2=(`ls ./`)
    [root@server shell01]# array3=(jack harry "Miss Hou" [5]=tom)
    
    查看普通数组信息:
    [root@server shell01]# declare -a
    declare -a array='([0]="var1" [1]="var2" [2]="var3")'
    declare -a array1='([0]="uu1" [1]="uu2" [2]="uu3" [3]="uu4")'
    declare -a array2='([0]="1.sh" [1]="2.sh" [2]="3.sh" [3]="4.sh" [4]="passwd")'
    declare -a array3='([0]="jack" [1]="harry" [2]="Miss Hou" [5]="tom")'
    [root@server shell01]# 
    [root@server shell01]# 
    [root@server shell01]# echo ${array[*]}
    var1 var2 var3
    [root@server shell01]# echo ${array[@]}
    var1 var2 var3
    [root@server shell01]# echo ${array[2]}
    var3
    [root@server shell01]# echo ${array2[@]}
    1.sh 2.sh 3.sh 4.sh passwd
    [root@server shell01]# echo ${array2[3]}
    4.sh
    [root@server shell01]# 
    [root@server shell01]# echo ${array2[*]2}
    3.sh 4.sh
    [root@server shell01]# echo ${#array2[*]}
    5
    [root@server shell01]# echo ${!array2[*]}
    0 1 2 3 4
    [root@server shell01]# echo ${!array3[*]}
    0 1 2 5
    
    
    关联数组定义:
    首先声明关联数组
    declare -A asso_array1
    declare -A asso_array2
    declare -A asso_array3
    
    数组赋值:
    一次赋一个值:
    数组名[索引|下标]=变量值
    [root@server ~]# asso_array1[linux]=one
    [root@server ~]# asso_array1[java]=two
    [root@server ~]# asso_array1[php]=three
    一次赋多个值:
    [root@server ~]# asso_array2=([name1]=harry [name2]=jack [name3]=amy [name4]="Miss Hou")
    查看关联数组:
    [root@server ~]# declare -A
    declare -A asso_array1='([php]="three" [java]="two" [linux]="one" )'
    declare -A asso_array2='([name3]="amy" [name2]="jack" [name1]="harry" [name4]="Miss Hou" )'
    
    [root@server ~]# echo ${asso_array1[linux]}
    one
    [root@server ~]# echo ${asso_array1[php]}
    three
    [root@server ~]# echo ${asso_array1[*]}
    three two one
    [root@server ~]# echo ${!asso_array1[*]}
    php java linux
    [root@server ~]# echo ${#asso_array1[*]}
    3
    [root@server ~]# echo ${#asso_array2[*]}
    4
    [root@server ~]# echo ${!asso_array2[*]}
    name3 name2 name1 name4
    
    
    9. 交互式定义变量的值 read    主要用于让用户去定义变量值
    -p 提示信息
    -n 字符数 (限制变量值的字符数)
    -s 不显示   
    -t 超时(默认单位秒)(限制用户输入变量值的超时时间)
    
    [root@MissHou shell01]# cat 1.txt 
    10.1.1.1 255.255.255.0
    
    [root@MissHou shell01]# read -p "Input your IP and Netmask:" ip mask < 1.txt 
    [root@MissHou shell01]# echo $ip
    10.1.1.1
    [root@MissHou shell01]# echo $mask
    255.255.255.0
    
    
    10. 其他变量(扩展)
    1)取出一个目录下的目录和文件:dirname和 basename 
    2)变量"内容"的删除和替换
    一个“%”代表从右往左去掉一个/key/
    两个“%%”代表从右往左最大去掉/key/
    一个“#”代表从左往右去掉一个/key/
    两个“##”代表从左往右最大去掉/key/
    
    # A=/root/Desktop/shell/mem.txt 
    # echo $A
    /root/Desktop/shell/mem.txt
    # dirname $A   取出目录
    /root/Desktop/shell
    # basename $A  取出文件
    mem.txt
    
    # url=www.taobao.com
    # echo ${#url}             获取变量的长度
    # echo ${url#*.}
    # echo ${url##*.}
    # echo ${url%.*}
    # echo ${url%%.*}
    
    ++++++++++++++++++++++++++++++++++++++++++++++++++
    以下内容自己完成:
    替换:/ 和 //
     1015  echo ${url/ao/AO}
     1017  echo ${url//ao/AO}   贪婪替换
    
    替代: - 和 :-  +和:+
     1019  echo ${abc-123}
     1020  abc=hello
     1021  echo ${abc-444}
     1022  echo $abc
     1024  abc=
     1025  echo ${abc-222}
    
    ${变量名-新的变量值} 或者 ${变量名=新的变量值}
    变量没有被赋值:会使用“新的变量值“ 替代
    变量有被赋值(包括空值): 不会被替代
    
     1062  echo ${ABC:-123}
     1063  ABC=HELLO
     1064  echo ${ABC:-123}
     1065  ABC=
     1066  echo ${ABC:-123}
    
    ${变量名:-新的变量值} 或者 ${变量名:=新的变量值}
    变量没有被赋值或者赋空值:会使用“新的变量值“ 替代
    变量有被赋值: 不会被替代
    
     1116  echo ${abc=123}
     1118  echo ${abc:=123}
    
    [root@server ~]# unset abc
    [root@server ~]# echo ${abc:+123}
    
    [root@server ~]# abc=hello
    [root@server ~]# echo ${abc:+123}
    123
    [root@server ~]# abc=
    [root@server ~]# echo ${abc:+123}
    
    ${变量名+新的变量值}
    变量没有被赋值或者赋空值:不会使用“新的变量值“ 替代
    变量有被赋值: 会被替代
    [root@server ~]# unset abc
    [root@server ~]# echo ${abc+123}
    
    [root@server ~]# abc=hello
    [root@server ~]# echo ${abc+123}
    123
    [root@server ~]# abc=
    [root@server ~]# echo ${abc+123}
    123
    ${变量名:+新的变量值}
    变量没有被赋值:不会使用“新的变量值“ 替代
    变量有被赋值(包括空值): 会被替代
    
    [root@server ~]# unset abc
    [root@server ~]# echo ${abc?123}
    -bash: abc: 123
    
    [root@server ~]# abc=hello
    [root@server ~]# echo ${abc?123}
    hello
    [root@server ~]# abc=
    [root@server ~]# echo ${abc?123}
    
    ${变量名?新的变量值}
    变量没有被赋值:提示错误信息
    变量被赋值(包括空值):不会使用“新的变量值“ 替代
    
    [root@server ~]# unset abc
    [root@server ~]# echo ${abc:?123}
    -bash: abc: 123
    [root@server ~]# abc=hello
    [root@server ~]# echo ${abc:?123}
    hello
    [root@server ~]# abc=
    [root@server ~]# echo ${abc:?123}
    -bash: abc: 123
    
    ${变量名:?新的变量值}
    变量没有被赋值或者赋空值时:提示错误信息
    变量被赋值:不会使用“新的变量值“ 替代
    
    说明:?主要是当变量没有赋值提示错误信息的,没有赋值功能
    

     

    简单的四则运算

    算术运算:默认情况下,shell就只能支持简单的==整数==运算

     

    + - * /  %(取模,求余数) 
    
    Bash shell 的算术运算有四种方式:
    1. 使用 $(( ))
    2. 使用$[ ]
    3. 使用 expr 外部程式
    4. 使用let 命令
    
    注意:
    n=1
    let n+=1  等价于let n=n+1
    
    
    
    思考:能不能用shell做小数运算?
    [root@server shell01]# echo 1+1.5|bc
    2.5
    
    
    i++ 和 ++i (了解)
    对变量的值的影响:
    [root@node1 ~]# i=1
    [root@node1 ~]# let i++
    [root@node1 ~]# echo $i
    2
    [root@node1 ~]# j=1
    [root@node1 ~]# let ++j
    [root@node1 ~]# echo $j
    2
    
    对表达式的值的影响:
    [root@node1 ~]# unset i j
    [root@node1 ~]# i=1;j=1
    [root@node1 ~]# let x=i++         先赋值,再运算
    [root@node1 ~]# let y=++j         先运算,再赋值
    [root@node1 ~]# echo $i
    2
    [root@node1 ~]# echo $j
    2
    [root@node1 ~]# echo $x
    1
    [root@node1 ~]# echo $y
    2
    

     

    总结:

     

    $(())  $[]
    expr 注意空格,*要进行转义 
    
    let n+=1  等价   let n=n+1
    let n=n**5        n有初值,然后求次幂
    
    i++    ++i 
    对变量本身没有影响(自己+1);
    表达式中有影响;i++ 先赋值再运算  ++i先运算再赋值
    let x=i++   let x=++i
    

     

    条件判断

    1. 语法格式

    格式1:==test== 条件表达式

    格式2:[ 条件表达式 ]

    格式3:[[ 条件表达式 ]]  支持正则 =~

    说明:

    man test去查看,很多的参数都用来进行条件判断

    2. 条件判断相关参数

    与文件存在与否的判断

     

    -e    是否存在   不管是文件还是目录,只要存在,条件就成立
    -f    是否为普通文件
    -d    是否为目录
    -S    socket
    -p    pipe
    -c    character
    -b    block
    -L    软link
    
    三种语法格式:
    test -e file                    只要文件存在条件为真
    [ -d /shell01/dir1 ]            判断目录是否存在,存在条件为真
    [ ! -d /shell01/dir1 ]        判断目录是否存在,不存在条件为真
    [[ -f /shell01/1.sh ]]        判断文件是否存在,并且是一个普通的文件
    
    -s 判断文件是否有内容(大小),非空文件条件满足
    说明:-s表示非空,! -s 表示空文件
    说明:1.sh文件里有内容的。
    [root@server shell01]# test -s 1.sh
    [root@server shell01]# echo $?
    0
    [root@server shell01]# touch aaa
    [root@server shell01]# cat aaa
    [root@server shell01]# test -s aaa
    [root@server shell01]# echo $?
    1
    [root@server shell01]# test ! -s aaa
    [root@server shell01]# echo $?
    0
    [root@server shell01]# test ! -s 1.sh
    [root@server shell01]# echo $?
    1
    

     

    文件权限相关的判断

     

    -r    当前用户对其是否可读
    -w    当前用户对其是否可写
    -x    当前用户对其是否可执行
    -u    是否有suid
    -g    是否sgid
    -k    是否有t位
    

     

    两个文件的比较判断

     

    file1 -nt  file2    比较file1是否比file2新    
    file1 -ot  file2     比较file1是否比file2旧
    file1 -ef  file2    比较是否为同一个文件,或者用于判断硬连接,是否指向同一个inode
    
    test file1 -nt file2    
    [ file1 -ot file2 ]
    

     

    整数之间的判断

     

    -eq    相等
    -ne    不等
    -gt    大于
    -lt    小于
    -ge   大于等于
    -le    小于等于
    

     

    字符串之间的判断

     

    -z  是否为空字符串           字符串长度为0,就成立
    -n  是否为非空字符串        只要字符串非空,就是成立
    string1 = string2         是否相等
    string1 != string2         不等
    
    [root@server shell01]# AAA=hello
    [root@server shell01]# BBB=world
    [root@server shell01]# test -z $AAA
    [root@server shell01]# echo $?
    1
    [root@server shell01]# test -n $AAA
    [root@server shell01]# echo $?
    0
    
    [root@server shell01]# [ $AAA = $BBB ]
    [root@server shell01]# echo $?
    1
    [root@server shell01]# [ $AAA != $BBB ]
    [root@server shell01]# echo $?
    0
    

     

    多重条件判断

     

    逻辑判断符号:
     -a     和 &&      (and 逻辑与)       两个条件同时满足,整个大条件为真
     -o     和 ||    (or 逻辑或)            两个条件满足任意一个,整个大条件为真
    
    
    [ 1 -eq 1 -a 1 -ne 0 ]                整个表达式为真
    [ 1 -eq 1 ] && [ 1 -ne 0 ]            
    
    
    [ 1 -eq 1 -o 1 -ne 1 ]                整个表达式为真
    [ 1 -eq 1 ] || [ 1 -ne 1 ]
    
    [root@server shell01]# [ 1 -eq 0 ] && echo true || echo false
    false
    [root@server shell01]# [ 1 -eq 1 ] && echo true || echo false
    true
    
    &&:前面的表达式为真
    ||:前面的表达式为假
    
    
    
    总结:
    1、; && ||都可以用来分割命令或者表达式
    2、; 完全不考虑前面的语句是否正确执行,都会执行;号后面的内容
    3、&& 需要考虑&&前面的语句的正确性,前面语句正确执行才会执行&&后的内容;反之亦然
    make && make install
    4、|| 需要考虑||前面的语句的非正确性,前面语句执行错误才会执行||后的内容;反之亦然
    5、如果&&和||一起出现,从左往右依次看,按照以上原则
    

     

    3. 示例

     

    示例:
    数值比较:
    [root@server ~]# [ $(id -u) -eq 0 ] && echo "the user is admin"
    [root@server ~]$ [ $(id -u) -ne 0 ] && echo "the user is not admin"
    [root@server ~]$ [ $(id -u) -eq 0 ] && echo "the user is admin" || echo "the user is not admin"
    
    [root@server ~]# uid=`id -u`
    [root@server ~]# test $uid -eq 0 && echo this is admin
    this is admin
    [root@server ~]# [ $(id -u) -ne 0 ]  || echo this is admin
    this is admin
    [root@server ~]# [ $(id -u) -eq 0 ]  && echo this is admin || echo this is not admin
    this is admin
    [root@server ~]# su - stu1
    [stu1@server ~]$ [ $(id -u) -eq 0 ]  && echo this is admin || echo this is not admin
    this is not admin
    [stu1@server ~]$ 
    
    
    类C风格的数值比较:
    注意:在(( ))中,=表示赋值;==表示判断
     1159  ((1==2));echo $?
     1160  ((1<2));echo $?
     1161  ((2>=1));echo $?
     1162  ((2!=1));echo $?
     1163  ((`id -u`==0));echo $?
    
     1209  ((a=123));echo $a
     1210  unset a
     1211  ((a==123));echo $?
    
    
    
    字符串比较:
    注意:双引号引起来,看作一个整体;= 和 == 在 [ 字符串 ] 比较中都表示判断
     1196  a='hello world';b=world
     1197  [ $a = $b ];echo $?
     1198  [ "$a" = "$b" ];echo $?
     1199  [ "$a" != "$b" ];echo $?
     1200  [ "$a" !== "$b" ];echo $?        错误
     1201  [ "$a" == "$b" ];echo $?
     1202  test "$a" != "$b";echo $?
    
    
    
    思考:[ ] 和 [[ ]] 有什么区别?
    
     1213  a=
     1214  test -z $a;echo $?
     1215  a=hello
     1216  test -z $a;echo $?
     1217  test -n $a;echo $?
     1217  test -n "$a";echo $?
    
    # [ '' = $a ];echo $?
    -bash: [: : unary operator expected
    2
    # [[ '' = $a ]];echo $?
    0
    
    
     1278  [ 1 -eq 0 -a 1 -ne 0 ];echo $?
     1279  [ 1 -eq 0 && 1 -ne 0 ];echo $?
     1280  [[ 1 -eq 0 && 1 -ne 0 ]];echo $?

     

      审核编辑:汤梓红
     
  • 打开APP阅读更多精彩内容
    声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

    全部0条评论

    快来发表一下你的评论吧 !

    ×
    20
    完善资料,
    赚取积分