设为首页收藏本站

17nyy 国内著名免费干货分享社区

 找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
热搜: 活动 交友 discuz
查看: 355|回复: 0

Shell 编程基础(一)

[复制链接]
  • TA的每日心情
    难过
    2020-10-4 20:24
  • 主题

    帖子

    0

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    0
    发表于 2020-1-1 21:39:30 | 显示全部楼层 |阅读模式
      Shell脚本的关键在于输入多个命令并处理每个命令的结果,甚至需要将一个命令的结果传给另一个命令。Shell可以让你将多个命令串起来,一次执行完成。
      基本格式:
      第一行必须为固定格式,指明脚本使用哪种shell来运行脚本,通常shell脚本中会以 # 作为注释,注释号后面的内容不会参与脚本的运行,但是,第一行是个例外。
      变量命名法则:
      1、不能使程序中的保留字:例如if, for
      2、只能使用数字、字母及下划线,且不能以数字开头,不能使用 - (减号)
      正确:_abc123 ;  abc123 ; abc_123
      错误:var1-abc=100  ;  var1-123=100  ;  -var1=100
      3、见名知义
      name  ,date
      4、统一命名规则:驼峰命名法
      HostName
      直接运行脚本的时候,会新开一个shell进程,脚本中默认关闭了alias功能
      局部变量:生效范围为当前shell进程;对当前shell之外的其它shell进程,包括当前shell的子shell进程均无效。
      环境(全局)变量:生效范围为当前shell进程及其子进程
      本地变量:生效范围为当前shell进程中某代码片断,通常指函数
      位置变量:$1, $2,$N ...来表示,用于让脚本在脚本代码中调用通过命令行传递给它的参数
      特殊变量:
      $?
      返回上一条命令执行状态
      $0
      命令本身
      $*
      传递给脚本的所有参数,全部参数合为一个字符串,用双括号括起
      $@
      传递给脚本的所有参数,每个参数为独立字符串,每个都用双括号括起
      $#
      传递给脚本的参数的个数
      $_
      上个命令的最后一个参数
      $$
      显示当前进程号
      $PPID
      显示父进程号
      $!
      上一个子进程的进程号
      $-
      在Shell启动或使用set命令时提供选项
      $n
      位置参数值,n表示位置
      使用pstree -p 查看进程树
      $@ $* 只在被双引号包起来的时候才会有差异
      变量赋值:
      变量名=值
      赋值等号与值之间没有空格
      [中括号里写变量的时候,记得加双引号]
      (1) 可以是直接字串; 变量名=root   
      如果值为带空格或特殊字符时,请使用双引号和  \  转义符号
      (2) 变量引用:     变量名="$USER"
      (3) 命令引用:    变量名=`指令`
      变量名=$(指令)
      变量引用:
      ${变量名}
      $变量名
      "双引号":弱引用,其中的变量引用会被替换为变量值
      '单引号':强引用,其中的变量引用不会被替换为变量值,而保持原字符串
      显示已定义的所有变量
      set
      删除变量:
      unset 变量名
      只读变量:只能声明,但不能修改和删除
      声明只读变量:
      readonly 变量名
      declare -r 变量名
      查看只读变量:
      readonly –p
      位置变量:在脚本代码中调用通过命令行传递给脚本的参数
      set -- 清空所有位置变量
      进程使用退出状态来报告成功或失败
      一般来讲,
      0 代表成功,1-255代表失败
      以下表为在未自定义退出状态码(exit)时的常见,可以参考
      状态码
      描述
      0
      命令成功结束
      1
      一般性未知错误,可能是无效参数
      2
      不适合的shell命令
      126
      命令不可执行
      127
      没有找到命令
      128
      无效的退出参数
      128+x
      与Linux信号x相关的严重错误
      130
      通过Ctrl+C终止的命令
      255
      正常范围之外的退出状态码
      bash自定义退出状态码
      exit [n]:自定义退出状态码
      脚本中一旦遇到exit命令,脚本会立即终止;终止退出状态取决于exit命令后面的数字
      如果未给脚本指定退出状态码,整个脚本的退出状态码取决于脚本中执行的最后一条命令的状态码
      退出状态可以使用变量作为值,退出状态码最大值为255,如果值超过255时,将会通过
      取模运算  值/256 把状态码缩减到0~255整数值之间,比如:
      exit 300
      300/256 取模=44    400/256 取模=144
      bash自带脚本测试
      bash -n 语法检查
      bash -x  脚本执行按步调试

      算术运算
      +
      两个整数相加

      %
      两整数相除,取余数
      -
      第一个数减去第二个数



      *
      两整数相乘



      \
      第一个整数除以第二个整数



      实现算术运算的方法:
      (1) let 变量名=算术表达式
      (2) var=$[算术表达式]
      sum=$[uid10+uid20]
      sum=$[$uid10+$uid20]
      (3) var=$((算术表达式))
      sum=$((uid10+uid20))
      (4) declare –i 变量名 = 数值
      (5) echo ‘算术表达式’ | bc
      

      条件测试
      若真,则返回0
      若假,则返回1
      1. test 语句
      2. [空格 表达式
      推荐使用第2种方法,因为我们去看系统本身自带的脚本编写使用的正是这一类方法。
      第一个 [ 之后和第二个 ] 之前必须加上一个空格,否则会报错
      
      上面的中括号是错误的格式,会报以下的错误
      
      ++++++++++++我是华丽的分界线++++++++++++++++++++
      
      上面的中括号是错误的格式,会报以下的错误
      
      
      ++++++++++++我是华丽的分界线++++++++++++++++++++
      以下才是正确的格式
      
      可以判断4类条件
      a.数值测试b.字符串测试c.文件测试d.布尔逻辑组合测试条件
      以下逐个看
      a.数值测试
      可以用在数字和变量(变量值是数字)上
      -lt
      小于
      less than

      -eq
      等于 equal
      -gt
      大于
      great than

      -ne
      不等于
      not equal
      -ge
      大于等于
      great equal



      -le
      小于等于
       less equal



      注意,当碰到系统特殊符号的时候,记得 \ 转义
      
      但是涉及到浮点值的时候,数值测试会有一个限制 ,我们来看一个例子:
      
      变量$var1的值是浮点值,脚本对这个值进行了测试,显然这里就报错。因此,请记住
      bash shell只能处理整数(zshell能很好的解决这个问题),当然,如果这个值只是用来
      echo 输出是没问题的。
      因此,此时shell的退出状态码也是为非0值了,也就是执行了else 之后的语句。
      b.字符串测试
      相等比较
      在比较字符的相等性时,会将所有的标点和大小写情况都考虑在内的
      ==
      字符串是否相同

      -n 字符串
      字符串是否为非0
      !=
      字符串是否不同

      -z 字符串
      字符串是否为0
      <
      左边的ascii码是否小于右边的ascii码

      =~
      左侧字符串是否能够被
      右侧的表达式所匹配
      用于[[ 双中括号中 ]]
      >
      左边的ascii码是否大于右边的ascii码



      


      使用调试工具来直观的看出状态码与结果
      使用=或者==  ,或者"" 和"空格" 的结果都是一致的

      使用单引号也是一致的
      这里需要注意的是,当值中带有空格的时候,变量没有双引号引起来会是报错的
      当变量双引号引起来的时候,
      正确输出
      所以,在写 bash 脚本的时候,别偷懒,对于变量的引用最好都加上双引号!就变量引用上来说,虽然 zsh 在这点要强过 bash,但是处于兼容性的考虑,还是把双引号带上吧。
      字符的比较
      先来一个初学者常犯的错误
      乍一看,没问题啊
      可以当然我们使用ls查看当前目录的时候,发现$var2的值被重定向至以$var2的值,也就是abcdA的文件中了。这是一个不易觉察的严重问题。脚本把 > 解析成了重定向输出符号。同时,因为重定向成功,因此exit退出码为0,就会执行then语句了。
      因此,使用 双引号 引起来 或 \ 转义解决吧。
      这才是规范的脚本写法,别掉坑了哦~~
      
      这才是正确的输出结果
      
      字符串的大小写比较
      
      在Shell的比较测试中,大写字母是被认为是小于小写字母的。这一点与sort命令恰好相反,同样的字符串用sort排序时,小写字母先出现,这是由于各个命令使用的排序技术不同造成的。
      因此,总结为2句话:
      shell编程时,小写字母 >(大于) 大写字母
      sort排序时,小写字母  >(优先) 大写字母
      如果出现以下大小写混合并相同的情况呢
      那么就从左边开始比,第一位a相同,那就比第一位,b > A  根据以前的总结,结果就是
      $var1 > $var2,不信就以图验证下
      而sort命令则是以按小写字母>相同的大写字母排序的,注意哦~
      如果是把数值的比较用了字符串的比较,那么你会怀疑你的数学是体育老师教的吧
      数字越大就是大
      最后的总结,如果使用错了操作符,可能无法得到正确的结果。
      比较类型
      使用的操作符
      数值
      -lt; -gt; -le; -ge; -eg; -ne
      字符串
      =; !=; >; <;
      检查变量是否含有数据
      -n 就是判断字符串是否为0
      
      结果
      ++++++++++++我是华丽的分界线++++++++++++++++++++
      结果
      ++++++++++++我是华丽的分界线++++++++++++++++++++
      结果
      ++++++++++++我是华丽的分界线++++++++++++++++++++
      结果,同时用双引号引起来的 空格 也是一样的结果
      ++++++++++++我是华丽的分界线++++++++++++++++++++
      结果,没有定义就是0,就是没有值
      ++++++++++++我是华丽的分界线++++++++++++++++++++
      -z 就是判断字符串是否为0
      
      结果,与-n选项的结果是一致的
      ++++++++++++我是华丽的分界线++++++++++++++++++++
      结果,用单引号引起来的 空格,结果也是一样的,与-n的结果是一致的
      ++++++++++++我是华丽的分界线++++++++++++++++++++
      结果,用单引号引起来的的什么也没有,结果也是一样的,与-n的结果是一致的
      ++++++++++++我是华丽的分界线++++++++++++++++++++
      结果,与-n 的结果是一致的

      最后的总结,不管是单双引号引用的没有值变量,或者没有被声明的变量,值都是0.也就是空。
      空和未初始化的变量会对shell脚本测试造成灾难性的影响,如果不是很确定一个变量的内容,最好在将其用于数值或字符串比较之前先通过- n 或 -z 来测试下变量是否含有值。
      最后最后的警示演示:
      清除下历史记录
      #history -c
      执行脚本,我的天啊,我的脚本原本的意图只是想删除/app/下面的全部文件而已,怎么变成了删除根目录下面的全部文件夹了呢!!太吓人了啊~~脚本有风险,且行且珍惜。哈哈~~
      c.文件测试
      用来测试Linux文件和目录的状态。
       (选项后接file名,比如 -e file  )

      存在性测试
      -e
      文件存在性测试,存在为真,否则为假

      存在性及类别测试
      -b
      是否存在且为块设备文件
      -c
      是否存在且为字符设备文件
      -d
      是否存在且为目录文件
      -f
      是否存在且为普通文件
      -L(大写)
      存在且为符号链接文件
      -p
      是否存在且为命名管道文件
      -S(大写)
      是否存在且为套接字文件

      文件权限测试
      -r
      是否存在且可读
      -w
      是否存在且可写
      -x
      是否存在且可执行

      文件特殊权限测试
      -u
      是否存在且拥有suid权限
      -g
      是否存在且拥有sgid权限
      -k
      是否存在且拥有sticky权限

      文件大小测试
      -s(小写)
      是否存在且非空

      文件是否打开测试
      -t fd
      fd文件描述符是否在某终端已经打开
      -N
      文件自从上一次被读取之后是否被修改过
      -O
      当前有效用户是否为文件属主
      -G
      当前有效用户是否为文件属组,只会检查默认的主组,而不检查附加组
      所谓的有效用户就是指,使用特殊的suid权限时,运行此程序的用户

      双文件测试
      File1 -ef File2
      File1是否是File2的硬链接
      File1 -nt File2
      File1是否新于File2(mtime)。必须确认文件是存在的,否则会返回错误结果
      File1 -ot File2
      File1是否旧于File2(mtime)。必须确认文件是存在的,否则会返回错误结果
      因为这些操作符比较统一规范,自己多练习就好了,我就只给2个例子:
      

      d.布尔逻辑组合测试条件
      &&
      并且(AND),操作符2边的条件同时必须满足才会执行后面的指令,2个条件组合成为真
      [[ 条件1 && 条件2 ]]
      test模式下使用-a
      ||
      或者(OR),操作符2边的条件只能满足其中一个才会执行后面的指令,2个条件组合成为假
      [[ 条件1 || 条件2 ]]
      test模式下使用-o
      !
      
      ! 条件
      第一种方式:双中括号格式
      如:[[ -r FILE ]] && [[ -w FILE ]]
      [[ -r FILE ]] || [[ -w FILE ]]
      第二种方式:必须使用测试命令test进行
      如:test [ -f /bin/cat -a -x /bin/cat ] && cat /etc/fstab
      test [ -f /bin/cat -o -x /bin/cat ] && cat /etc/fstab
      高级特性
      bash shell 提供了2项可在if-then语句中使用的高级特性:
      a.用于数学表达式的双括号 ((  ))
      不需有将双括号中表达式里的运算符转义
      建议括号内2边加空格
      ((空格 表达式  空格))
      b.用于高级字符串处理功能的双中括号 [[  ]]
      不是所有的shell都支持双中括号
      建议双中括号内2边加空格
      [[空格 表达式  空格]]
      操作符号
      va1++
      先运算,后自增1

      !
      逻辑求反
      va1--
      先运算,后自减1

      ~
      位求反
      ++va1
      先自加1,后运算

      **
      幂运算
      --va1
      先自减1,后运算

      <<
      左位移
      &
      位布尔和

      >>
      右位移
      |
      位布尔或

      &&
      逻辑和
      =~
      双中括号中后面跟
      扩展正则表达式

      ||
      逻辑或
      va1++  (va1--)
      自身先和后面的运算式运算,表达式的结果跟自身没有任何关系,再自身+-1的值赋与以后使用。
      例子: a=5 b=2
      a++ +b=7   (a+b=7,此时a的值为6)
      上条表达式后的下一条表达式a的值已经变化了
      a+b=8
      ++id ( --id)
      例子: a=5 b=2
      ++a +b =8  (a+1=6,6+b=8,此时a的值为6)
      上条表达式后的下一条表达式a的值已经变化了
      a+b=8
      自身先+-1,再和后面的运算式运算,表达式的结果跟自身没有任何关系,
      以后的值就是原先自身先+-1
      双括号范例
      双中括号范例,扩展正则表达式
      双中括号范例,扩展正则表达式
      如果写成双等号的话,双中括号会自动转化为带转义的写法
      注意红框的地方
      第一章节到此结束,本人水平有限,如有错漏地方,欢迎指正。



    来源:51CTO技术博客https://blog.51cto.com/191226139/2044393
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|Archiver|手机版|小黑屋|一起赢 ( 蜀ICP备19039281号-1 )

    GMT+8, 2020-10-29 01:26 , Processed in 0.120611 second(s), 26 queries .

    Powered by Discuz! X3.4

    © 2001-2017 Comsenz Inc.

    快速回复 返回顶部 返回列表