• shell编程上


    1.1  前言

    1.1.1  为什么学Shell

    Shell脚本语言是实现Linux/UNIX系统管理及自动化运维所必备的重要工具, Linux/UNIX系统的底层及基础应用软件的核心大都涉及Shell脚本的内容。每一个合格 的Linux系统管理员或运维工程师,都需要能够熟练地编写Shell脚本语言,并能够阅 读系统及各类软件附带的Shell脚本内容。只有这样才能提升运维人员的工作效率,适 应曰益复杂的工作环境,减少不必要的重复工作,从而为个人的职场发展奠定较好的基础

    1.1.2  什么是shell

    Shell是一个命令解释器,它在操作系统的最外层,负责直接与用户对话,把用户的输入解释给操作系统,并处理各种各样的操作系统的输出结果,输出屏幕返回给用户。

    这种对话方式可以是:

    交互的方式:从键盘输入命令,通过/bin/bash的解析,可以立即得到Shell的回应

    [root@clsn ~]# ls            
    anaconda-ks.cfg  
    [root@clsn ~]# echo ls |bash 
    anaconda-ks.cfg   

    非交互的方式: 脚本

     

    1.1.3 什么是Shell脚本

      命令、变量和流程控制语句等有机的结合起来

             shell脚本擅长处理纯文本类型的数据,而linux中,几乎所有的配置文件,日志,都是纯文本类型文件

    1.1.4 脚本语言的种类

    一、编译型语言

    定义:指用专用的编译器,针对特定的操作平台(操作系统)将某种高级语言源代码一次性翻译成可被硬件平台直接运行的二进制机器码(具有操作数,指令、及相应的格式),这个过程叫做编译(./configure  make makeinstall );编译好的可执行性文件(.exe),可在相对应的平台上运行(移植性差,但运行效率高)。。

    典型的编译型语言有, C语言、C++等。

    另外,Java语言是一门很特殊的语言,Java程序需要进行编译步骤,但并不会生成特定平台的二进制机器码,它编译后生成的是一种与平台无关的字节码文件(*.class)(移植性好的原因),这种字节码自然不能被平台直接执行,运行时需要由解释器解释成相应平台的二进制机器码文件;大多数人认为Java是一种编译型语言,但我们说Java即是编译型语言,也是解释型语言也并没有错。

    二、解释型语言

    定义:指用专门解释器对源程序逐行解释成特定平台的机器码并立即执行的语言;相当于把编译型语言的编译链接过程混到一起同时完成的。

    解释型语言执行效率较低,且不能脱离解释器运行,但它的跨平台型比较容易,只需提供特定解释器即可。

    常见的解释型语言有, Python(同时是脚本语言)与Ruby等。

    三、脚本语言

    定义:为了缩短传统的编写–编译–链接–运行(edit-compile-link-run)过程而创建的计算机编程语言。

    特点:程序代码即是最终的执行文件,只是这个过程需要解释器的参与,所以说脚本语言与解释型语言有很大的联系。脚本语言通常是被解释执行的,而且程序是文本文件。

    典型的脚本语言有,JavaScript,Python,shell等。

    其他常用的脚本语句种类

    PHP是网页程序,也是脚本语言。是一款更专注于web页面开发(前端展示)的脚本语言,例如:Dedecms,discuz。PHP程序也可以处理系统日志,配置文件等,php也可以调用系统命令。

    Perl脚本语言。比shell脚本强大很多,语法灵活、复杂,实现方式很多,不易读,团队协作困难,但仍不失为很好的脚本语言,存世大量的程序软件。MHA高可用Perl写的

    Python,不但可以做脚本程序开发,也可以实现web程序以及软件的开发。近两年越来越多的公司都会要求会Python。

    Shell脚本与php/perl/python语言的区别和优势?

    shell脚本的优势在于处理操作系统底层的业务 (linux系统内部的应用都是shell脚本完成)因为有大量的linux系统命令为它做支撑。2000多个命令都是shell脚本编程的有力支撑,特别是grep、awk、sed等。例如:一键软件安装、优化、监控报警脚本,常规的业务应用,shell开发更简单快速,符合运维的简单、易用、高效原则.

      PHP、Python优势在于开发运维工具以及web界面的管理工具,web业务的开发等。处理一键软件安装、优化,报警脚本。常规业务的应用等php/python也是能够做到的。但是开发效率和复杂比用shell就差很多了。

    系统环境说明

    [root@clsn scripts]# cat /etc/redhat-release 
    CentOS Linux release 7.4.1708 (Core) 
    [root@clsn scripts]# uname -r
    3.10.0-693.el7.x86_64
    [root@clsn scripts]# getenforce 
    Disabled
    [root@clsn scripts]# systemctl status firewalld.service 
    ● firewalld.service - firewalld - dynamic firewall daemon
       Loaded: loaded (/usr/lib/systemd/system/firewalld.service; disabled; vendor preset: enabled)
       Active: inactive (dead)
         Docs: man:firewalld(1)

    1.1.5 系统中的shell

    查看系统中的命解释器

    [root@clsn ~]# cat /etc/shells 
    /bin/sh
    /bin/bash
    /sbin/nologin
    /usr/bin/sh
    /usr/bin/bash
    /usr/sbin/nologin

    常用操作系统的默认shell

    1.Linux是Bourne Again shell(bash)

    2.Solaris和FreeBSD缺省的是Bourne shell(sh)

    3.AIX下是Korn Shell(ksh)

    4.HP-UX缺省的是POSIX shell(sh)

    [root@clsn ~]# echo $SHELL
    /bin/bash

    bash版本

    [root@clsn scripts]# bash -version
    GNU bash, 版本 4.2.46(2)-release (x86_64-redhat-linux-gnu)
    Copyright (C) 2011 Free Software Foundation, Inc.
    许可证 GPLv3+: GNU GPL 许可证版本3或者更高 
     
    这是自由软件,您可以自由地更改和重新发布。
    在法律允许的范围内没有担保. 

    bash 破壳漏洞

    使用 命令 env x='() { :;}; echo be careful' bash -c "echo this is a test"
    如果返回结果为一行,则为正常,
     [root@clsn ~]# env x='() { :;}; echo be careful' bash -c "echo this is a test"
     this is a test
     
    #解决办法 升级当前的bash版本
     yum install update bash

    shbash 的关系

    [root@clsn ~]#  ll /bin/sh
    lrwxrwxrwx. 1 root root 4 11月 13 11:15 /bin/sh -> bash

    /bin /user/bin 的关系

    [root@clsn ~]# ll /bin -d
    lrwxrwxrwx. 1 root root 7 11月 13 11:15 /bin -> usr/bin

    1.2 脚本书写规范

    1.2.1 脚本统一存放目录

    [root@clsn ~]# mkdir -p /server/scripts/
    [root@clsn ~]# cd /server/scripts/

    1.2.2 选择解释器

    注意格式 ↓

    其中开头的“#!”字符又称为幻数,在执行bash脚本的时候,内核会根据“#!”后的解释器来确定该用那个程序解释这个脚本中的内容。

    [root@clsn scripts]# head -1 /etc/init.d/*
    ==> /etc/init.d/functions <==
    # -*-Shell-script-*-
     
    ==> /etc/init.d/netconsole <==
    #!/bin/bash
     
    ==> /etc/init.d/network <==
    #! /bin/bash

    1.2.3 编辑脚本使用vim

    使用 .vimrc 文件,能够快速的生成开头的注释信息

    [root@clsn scripts]# cat  ~/.vimrc 
    autocmd BufNewFile *.py,*.cc,*.sh,*.java exec ":call SetTitle()"
     
    func SetTitle()
        if expand("%:e") == 'sh'
            call setline(1,"#!/bin/bash")
            call setline(2, "##############################################################")
            call setline(3, "# File Name: ".expand("%"))
            call setline(4, "# Version: V1.0")
            call setline(5, "# Author: clsn")
            call setline(6, "# Organization: http://blog.znix.top")
            call setline(7, "# Created Time : ".strftime("%F %T"))
            call setline(8, "# Description:")
            call setline(9, "##############################################################")
            call setline(10, "")
        endif
    endfunc

    使用后的效果

    [root@clsn scripts]# cat  scripts_test.sh 
    #!/bin/bash
    ##############################################################
    # File Name: scripts_test.sh
    # Version: V1.0
    # Author: clsn
    # Organization: http://blog.znix.top
    # Created Time : 2017-12-04 11:39:57
    # Description:  First scripts file
    ##############################################################

    在Shell脚本中,跟在#后面的内容表示注释。注释部分不会被执行,仅给人看。注释可以自成一行,也可以跟在命令后面,与命令同行。要养成写注释的习惯,方便自己与他人。

    最好不用中文注释,因为在不同字符集的系统会出现乱码。(字符集为zh_CN.UTF-8,为中文)。

    1.2.4 多行注释

    [root@web01 ~]# cat oldboy.sh

    #!/bin/bash

    :<<EOF

    echo "I am oldboy"

    echo "I am oldboy"

    echo "I am oldboy"

    EOF 

    1.2.5 文件名规范

             名字要有意义,并且结尾以 .sh 结束

    1.2.6 开发的规范和习惯小结

    1) 放在统一的目录

    2) 脚本以.sh为扩展名

    3) 开头指定脚本解释器。

    4) 开头加版本版权等信息,可配置~/.vimrc文件自动添加。

    5) 脚本不要用中文注释,尽量用英文注释。

    6) 代码书写优秀习惯

      a、成对的内容一次性写出来,防止遗漏,如[  ]、‘ ‘、” “等

      b、[  ]两端要有空格,先输入[  ],退格,输入2个空格,再退格写。

      c、流程控制语句一次书写完,再添加内容。(if 条件 ; then  内容;fi)ddd

      d、通过缩进让代码易读。

      f、脚本中的引号都是英文状态下的引号,其他字符也是英文状态。

    1.3 shell脚本的执行

    1.3.1 执行脚本的办法

    sh/bash   scripts.sh 
    chown +x   ./scripts.sh  && ./scripts.sh  
    source scripts.sh
    . (空格) scripts.sh
    cat oldboyedu.sh |bash  # 效率较低

    source  . (点) 的作用

    soucre命令

    [root@clsn ~]# help source  |head -2
    source: source 文件名 [参数]
        在当前 shell 中执行一个文件中的命令。

    . ()

    [root@clsn scripts]# help . |head -2
    .: . 文件名 [参数]
        在当前 shell 中执行一个文件中的命令。

    1.3.2 sh 于 source的区别

     

    [root@clsn scripts]# sh  clsn_test.sh 
    Hello World!
    [root@clsn scripts]# echo $clsn
    #  sh  新建一个Shell窗口(新建一个进程)执行一个文件中的命令。
     
    [root@clsn scripts]# source clsn_test.sh 
    Hello World!
    [root@clsn scripts]# echo $clsn
    Hello World!

    面试题:

    问sh test.sh后echo $user返回的结果__?

    [root@oldboy scripts]# cat test.sh 
    #!/bin/bash
    user=`whoami`

    1.4 Shell的变量

    1.4.1 什么是变量

    变量可以分为两类:环境变量(全局变量)和普通变量(局部变量)

      环境变量也可称为全局变量,可以在创建他们的Shell及其派生出来的任意子进程shell中使用,环境变量又可分为自定义环境变量和Bash内置的环境变量

      普通变量也可称为局部变量,只能在创建他们的Shell函数或Shell脚本中使用。普通变量一般是由开发者用户开发脚本程序时创建的。

             特殊变量

    1.4.2 环境变量

    使用 env/declare/set/export -p 命令查看系统中的环境变量,这三个命令的的输出方式稍有不同。

    [root@clsn scripts]# env
    XDG_SESSION_ID=1
    HOSTNAME=clsn
    TERM=linux
    SHELL=/bin/bash
    HISTSIZE=1000
    SSH_CLIENT=10.0.0.1 5537 22
    SSH_TTY=/dev/pts/0
    USER=root
    ~~~

             输出一个系统中的 环境变量

    [root@clsn ~]# echo $LANG
    zh_CN.UTF-8

    1.4.3 export命令

    [root@clsn ~]# help export 
    export: export [-fn] [名称[=值] ...] 或 export -p
    为 shell 变量设定导出属性。
     
    标记每个 NAME 名称为自动导出到后续命令执行的环境。如果提供了 VALUE
    则导出前将 VALUE 作为赋值。

    export命令的说明

    当前shell窗口及子shell窗口生效

    在新开的shell窗口不会生效,生效需要写入配置文件

    # 定义变量

     [root@clsn scripts]# export CSLN1=1

    # 当前窗口查看

     [root@clsn scripts]# echo $CSLN1
    1
    Unset   取消环境变量
      Unset  $s

    1.4.4 环境变量相关配置文件

    /etc/proflie

    /etc/bashrc

    ~/.bashrc

    ~/.bash_profile

    /etc/proflie.d/  # 目录

    四文件读取顺序(CentOS67都一样)

    ① /etc/profile  全用户

    ② ~/.bash_profile   当前用户的

    ③ ~/.bashrc  当前用户的

    ④ /etc/bashrc   最后 生效的   全用户

     

    文件读取过程示意图

    1.4.5 环境变量的知识小结

    ü 变量名通常要大写。

    ü 变量可以在自身的Shell及子Shell中使用。

    ü 常用export来定义环境变量。

    ü 执行env默认可以显示所有的环境变量名称及对应的值。

    ü 输出时用“$变量名”,取消时用“unset变量名”。

    ü 书写crond定时任务时要注意,脚本要用到的环境变量最好先在所执行的Shell脚本中重新定义。

    ü 如果希望环境变量永久生效,则可以将其放在用户环境变量文件或全局环境变量文件里。

    1.4.6 普通变量

     
    变量名:
     字母 数字 下划线,用字母开头,例如:oldboy_1
     规范写法:切记,不要使用没有意义的变量名,应该见名知意。
     oldboy_age
     OldboyAge
     oldboyAge 驼峰语法
    赋值符号 =,相等==
    变量内容:
    不加引号:会把变量及命令进行解析。适合定义整数,例如:a=123。
    双引号:  会把变量及命令进行解析,把变量内容做为一个整体,默认非整数定义用双引号。
    单引号:所见即所得。
    反引号:
    变量内容希望是一个命令等的结果的定义方法。
    [root@db03 scripts]# a=`pwd`
    [root@db03 scripts]# echo $a
    /server/scripts
    [root@db03 scripts]# a=$(pwd)
    [root@db03 scripts]# echo $a
    /server/scripts

    1.5  特殊变量

    1.5.1 位置变量

    常用的特殊位置参数说明

    位置变量

    作用说明

    $0

    获取当前执行的shell脚本的文件名,如果执行脚本带路径那么就包括脚本路径。

    $n

    获取当前执行的shell脚本的第n个参数值,n=1..9,当n为0时表示脚本的文件名,如果n大于9用大括号括起来{10},参数以空格隔开。

    $#

    获取当前执行的shell脚本后面接的参数的总个数

    $*

    获取当前shell的所有传参的参数,不加引号同$@;如果给$*加上双引号,例如: “$*”,则表示将所有的参数视为单个字符串,相当于“$1$2$3”。   一行

    $@

    获取当前shell的所有传参的参数,不加引号同$*;如果给$@加上双引号,例如: “$@”,则表示将所有参数视为不同的独立字符串,相当于“$1” “$2” “$3” “……”,这是将参数传递给其他程序的最佳方式,因为他会保留所有内嵌在每个参数里的任何空白。

    当“$*”和“$@”都加双引号时,两者有区别,都不加双引号时,两者无区别。  多行

    $0,$1.$2 ~ 参数实践

     

    $# 参数实践

    cat  10.sh

    echo $#

    sh  10.sh  {a..z}

    26

    $* 参数实践

    [root@clsn scripts]# vim 10.sh  
    echo $*
    sh 10.sh   {a..z}
    a b c d e f g h i j k l m n o p q r s t u v w x y z

    $*  $@ 对比实践

    [root@clsn scripts]# set -- "I am" handsome boy..
    [root@clsn scripts]# echo $1
    I am
    [root@clsn scripts]# echo $2
    handsome
    [root@clsn scripts]# echo $3
    boy..
    [root@clsn scripts]# echo $*
    I am handsome boy..
    [root@clsn scripts]# echo $@
    I am handsome boy..
    [root@clsn scripts]# for i in "$@";do echo $i ;done
    I am
    handsome
    boy..
    [root@clsn scripts]# for i in "$*";do echo $i ;done
    I am handsome boy..

    1.5.2 进程状态变量

    Shell进程的特殊状态变量说明

    位置变量

    作用说明

    $?

    获取执行上一个指令的执行状态返回值(0为成功,非零为失败),这个变量最常用

    $$

    获取当前执行的Shell脚本的进程号(PID),这个变量不常用,了解即可

    $!

    获取上一个在后台工作的进程的进程号(PID),这个变量不常用,了解即可

    $_

    获取在此之前执行的命令或脚本的最后一个参数,这个变量不常用,了解即可

    进程参数实践

    [root@clsn scripts]# echo $?
    0
    [root@clsn scripts]# echo $$
    1368
    [root@clsn scripts]# echo $!
     
    [root@clsn scripts]# echo $_
     
     [root@DB3 scripts]# echo 112222|tee  a.log
    112222   既能输出  又能追加

    1.5.3 echo参数说明

    参数

    参数说明

    -n

    不要追加换行

    -e

    启用下列反斜杠转义的解释

    -E

    显式地抑制对于反斜杠转义的解释

    `echo’ 对下列反斜杠字符进行转义:

    换行

    回车

    横向制表符

    

    退格

    v

    纵向制表符

    c

    抑制更多的输出

    1.6 变量的子串

    1.6.1 变量子串说明

    表达式

    说明

    ${parameter}

    返回变量$parameter的内容

    ${#parameter}

    返回变内容的长度(按字符),也适用于特殊变量

    ${parameteri:offset}

    在变量${parameter}中,从位置offset之后开始提取子串到结尾

    ${parameter:offset:length}

    在变量${parameter}中,从位置offset之后开始提取长度为length的子串

    ${parameter#word}

    从变量${parameter}开头开始删除最短匹配的word子串

    ${parameter##word}

    从变量${parameter}开头开始删除最长匹配的word子串

    ${parameter%word}

    从变量${parameter}结尾开始删除最短匹配的word子串

    ${parameter%%word}

    从变量${parameter}结尾开始删除最长匹配的word子串

    ${parameter/pattem/string}

    使用string代替第一个匹配的pattern

    ${parameter//pattem/string}

    使用string代替所有匹配的pattern

    计算变赋值的长度

    [root@DB3 scripts]# oldboy="I am oldboy"
    [root@DB3 scripts]# echo   ${#oldboy} 
    11
    [root@DB3 scripts]# echo $oldboy|wc -L
    11

    截取变量中的字符

     [root@db03 scripts]# echo ${oldboy:2}
    am oldboy
    [root@db03 scripts]# echo ${oldboy}
    I am oldboy
    [root@db03 scripts]# echo ${oldboy:2:2}
    am
    [root@db03 scripts]# echo ${oldboy:2:4}
    am o

    删除字符串

     [root@DB3 scripts]# OLDBOY=abcABC123ABCabc

    [root@DB3 scripts]# echo ${OLDBOY#a*C}

    123ABCabc

    [root@DB3 scripts]# echo ${OLDBOY##a*C}

    Abc

    [root@DB3 scripts]# echo ${OLDBOY%a*c}

    abcABC123ABC

    [root@DB3 scripts]# echo ${OLDBOY%%a*c}

     空

    有关上述匹配删除的小结

    #表示从幵头删除匹配最短。

    ##表示从开头删除匹配最长。

    %表示从结尾删除匹配最短。

    %%表示从结尾删除匹配最长。

    a*c表示匹配的突符串,*表示匹配所有,a*c匹配开头为a、中间为任意多个字符、结尾为c的字符串。

    a*C表示匹配的字符串,*表示匹配所有,a*C匹配开头为a、中间为任意多个字符、结尾为C的字符串。

    有关替换的小结

    替换变量内容

    [root@db03 scripts]# oldboy="I am oldboy,oldboy"

    [root@db03 scripts]# echo ${oldboy/oldboy/oldgirl}

    I am oldgirl,oldboy

    [root@db03 scripts]# echo ${oldboy//oldboy/oldgirl}

    I am oldgirl,oldgirl

    一个“/”表示替换匹配的第–个字符串。

    两个“/”表示替换匹配的所有字符串。

    1.6.2 Shell的特殊扩展变量说明

    表达式

    说明

    ${parameter:-word}

    如果parameter的变量值为空或未赋值,则会返回word字符串并替代变量的值用途.如果变量未定义,则返回备用的值,防止变量为空值或因未定义而导致异常

    ${parameter:=word}

    如果parameter的变量值为空或未赋值,则设置这个变量值为word,并返回其值。位置变量和特殊变量不适用用途:基本同上一个${parameter>word},但该变量又额外给parameter变量赋值了

    ${parameter:?word}

    如果parameter变量值为空或未赋值,那么word字符串将被作为标准错误输出,否则输出变量的值。用途:用于捕捉由于变量未定义而导致的错误,并退出程序

    ${parameter:+word}

    如果parameter变量值为空或未赋值,则什么都不做,否则word字符串将替代变量的值

    特殊变量实践

    脚本内容

    1.
    [root@db03 scripts]# echo $test
    [root@db03 scripts]# result=${test:-UNSET}
    [root@db03 scripts]# echo $result
    UNSET
    [root@db03 scripts]# result=${test-oldboy}
    [root@db03 scripts]# echo $result
    Oldboy
    2.
    [root@db03 scripts]# echo $test0
     
    [root@db03 scripts]# r=${test0:=oldgirl}
    [root@db03 scripts]# echo $test0
    oldgirl
    [root@db03 scripts]# echo $r
    Oldgirl
    3.

    [root@db03 scripts]# r=${test11:?oldgirl}
    -bash: test11: oldgirl
    [root@db03 scripts]# test11=11
    [root@db03 scripts]# r=${test11:?oldgirl}

     

    1.7 变量的数值计算

    1.7.1 仅支持整数的运算

    echo $((数学运算表达式))

    # 形式一
    [root@clsn scripts]# echo $((1 + 1))
    2
    [root@clsn scripts]# echo $((2*7-3/6+5))
    19
    # 形式二
    [root@clsn scripts]# ((clsn=2*8))
    [root@clsn scripts]# echo $clsn
    16
    # 形式三
    [root@clsn scripts]# znix=$((2*7-3/6+5))
    [root@clsn scripts]# echo $znix
    19

    延伸产物(重要)

    i++ 自增1

    i– 自减1

    ++i

    –i

             示例:

    [root@clsn scripts]# i=1
    [root@clsn scripts]# echo $((i++))
    1
    [root@clsn scripts]# echo $((i++))
    2
    [root@clsn scripts]# echo $((i--))
    3
    [root@clsn scripts]# echo $((i--))
    2
    [root@clsn scripts]# echo $((i--))
    1
    [root@clsn scripts]# echo $((++i))
    1
    [root@clsn scripts]# echo $((++i))
    2
    [root@clsn scripts]# echo $((++i))
    3
    [root@clsn scripts]# echo $((--i))
    2
    [root@clsn scripts]# echo $((--i))
    1
    [root@clsn scripts]# echo $((--i))
    0

    记忆方法:++,—

    变量a在前,表达式的值为a,然后a自增或自减,变量a在符号后,表达式值自增或自减,然后a值自增或自减。

    let命令

    [root@clsn scripts]# i=1
    [root@clsn scripts]# i=i+1
    [root@clsn scripts]# echo $i
    i+1
     
    [root@clsn scripts]# i=1
    [root@clsn scripts]# let i=i+1
    [root@clsn scripts]# echo $i
    2

    expr 命令

    1.整数计算

    2.判断扩展名

    3.判断输入是否为整数,非整数返回值为2

    4.计算变量的长度

    [root@clsn scripts]# expr 1+1
    1+1
    [root@clsn scripts]# expr 1 + 1
    2
    [root@clsn scripts]# expr 1 * 1
    expr: 语法错误
    [root@clsn scripts]# expr 1 * 1
    1

    非整数返回值为示例:

    [root@clsn scripts]# expr 1 + 1 
    2
    [root@clsn scripts]# echo $?
    0
    [root@clsn scripts]# expr -1 + 1 
    0
    [root@clsn scripts]# echo $?
    1
    [root@clsn scripts]# expr a + 1 
    expr: 非整数参数
    [root@clsn scripts]# echo $?
    2

    $[]运算符

    [root@clsn scripts]# echo $[1+2]
    3
    [root@clsn scripts]# echo $[1-2]
    -1
    [root@clsn scripts]# echo $[1*2]
    2
    [root@clsn scripts]# echo $[1/2]
    0

    typeset命令进行运算

    [root@clsn scripts]# typeset -i A=2017 B=2018
    [root@clsn scripts]# A=A+B
    [root@clsn scripts]# echo $A
    4035

    1.7.2 可以进行小数运算的命令

    bc    命令

    # 安装 bc  依赖于base源
    [root@clsn scripts]# yum -y install bc

      交互模式测试bc命令

    [root@clsn scripts]# bc
    bc 1.06.95
    Copyright 1991-1994, 1997, 1998, 2000, 2004, 2006 Free Software Foundation, Inc. 
    1+1
    2
    [root@clsn scripts]# echo 1+1.1|bc
    2.1

    免交互模式测试bc命令

    [root@clsn scripts]# echo 'scale=6;1/3'|bc
    .333333

    python 命令

    [root@clsn scripts]#  file `which yum `
    /usr/bin/yum: Python script, ASCII text executable
    [root@clsn scripts]# python 
    >>> import os
    >>> os.system('df -h')
    >>> 1+1.1
    2.1
    >>>exit()

    awk 命令

    [root@clsn ~]# echo "7.7 3.8"|awk '{print ($1-$2)}'
    3.9
    [root@clsn ~]# echo "358 113"|awk '{print ($1-3)/$2}'
    3.14159
    [root@clsn ~]# echo "3 9"|awk '{print ($1+3)*$2}'
    54
    [root@backup scripts]# awk BEGIN'{print 1.2+3.3}'
    4.5

    1.8 定义变量的方式

    1.8.1 三种定义变量的方式

      1、直接赋值

      2、传参 (传递参数)

      3、交互式设置变量,使用read命令

    1.8.2 read命令说明

    在命令行中使用

    [root@db03 scripts]# read -t 5 -p "请输入一个数字:"
    请输入一个数字:
    [root@db03 scripts]# read -t 5 -p "请输入一个数字:" a
    请输入一个数字:11
    [root@db03 scripts]# echo $a
    11

    read命令的帮助说明

    [root@clsn scripts]# read --help
    -bash: read: --: 无效选项
    read: 用法:read [-ers] [-a 数组] [-d 分隔符] [-i 缓冲区文字] [-n 读取字符数] [-N 读取字符数] [-p 提示符] [-t 超时] [-u 文件描述符] [-s不显示终端的任何输入] [名称 ...]

    1.8.3 写一个交互脚本,实现能够定义主机名及IP地址

    脚本内容

    [root@clsn scripts]# cat xiugaizhuji.sh 
    #!/bin/bash
    ethFile=/etc/sysconfig/network-scripts/ifcfg-eth[01]
    Now_eth=`hostname -I|awk -F "[. ]+" '{print $4}'`
     
    read -p "请输入主机名:" Hostname
    read -p "请输入IP地址的主机位:" HostIP
     
    hostnamectl set-hostname  $Hostname
     
    sed  -i "s#${Now_eth}#${HostIP}#g" $ethFile 
     
    read -p "是否重启服务器:{yes/no}"  REboot
     
    if [ $REboot == yes ]
    then 
       echo "系统将在10秒后重启!"
       shutdown -r  10 
    else
       echo "请稍后手动重启系统!"
    fi

    脚本测试结果

    [root@clsn scripts]# sh xiugaizhuji.sh
    请输入主机名:clsn
    请输入IP地址的主机位:180
    是否重启服务器:{yes/no}yes
    系统将在10秒后重启!
    [root@clsn scripts]# sh xiugaizhuji.sh
    请输入主机名:clsn
    请输入IP地址的主机位:180
    是否重启服务器:{yes/no}no
    请稍后手动重启!

    1.9 运算相关练习题

    1.9.1 实现一个加减乘除等功能的计算器

    实现脚本:

    [root@clsn scripts]# cat jishuanqi.sh 
    #!/bin/bash
    read -p "请输入第一个整数:" a
    read -p "请输入第二个整数:" b
    echo $a + $b =$(($a+$b))
    echo $a - $b =$(($a-$b))
    echo $a * $b =$(($a*$b))
    echo $a / $b =$(($a/$b))

             脚本执行过程:

    [root@clsn scripts]# sh jishuanqi.sh 
    请输入第一个整数:12
    请输入第二个整数:12
    12 + 12 =24
    12 - 12 =0
    12 * 12 =144
    12 / 12 =1

    精简方法

    [root@clsn scripts]# vim jishuanqi2.sh 
    #!/bin/bash
    echo $(($1))

             脚本执行过程:

    [root@clsn scripts]# sh jishuanqi2.sh  1+1
    2
    [root@clsn scripts]# sh jishuanqi2.sh  1*9
    9

    1.9.2 打印结果1+2+3+4+5+6+7+8+9+10=55

    脚本内容

    [root@clsn scripts]# vim yunshuan.sh
    #!/bin/bash
    Num=`seq -s + 1 10`
    echo  $Num=$(($Num))

    脚本执行结果

    [root@clsn scripts]# sh  yunshuan.sh
    1+2+3+4+5+6+7+8+9+10=55

    1.9.3 判断整数。

    [root@db03 scripts]# cat judge1.sh 
    #!/bin/bash
    expr 1 + $1 &>/dev/null
    if [ $? -eq 0 ]
    then
        echo "$1 是整数"
    else
        echo "$1 不是整数"
    fi
     
    [root@db03 scripts]# sh judge1.sh oldboy
    oldboy 不是整数
    [root@db03 scripts]# sh judge1.sh 123456
    123456 是整数
     
     

    1.9.4 expr计算字符串长度

    [root@db03 scripts]# oldboy="I am oldboy"
    [root@db03 scripts]# expr length "$oldboy"
    11
    [root@DB3 scripts]# oldboy="I am oldboy"
    [root@DB3 scripts]# echo   ${#oldboy} 
    11
    [root@DB3 scripts]# echo $oldboy|wc -L
    11
     

    1.9.5 expr判断文件扩展名

     
    [root@db03 scripts]# cat expr1.sh
    #!/bin/bash
    expr "$1" : ".*.txt" &>/dev/null
    if [ $? -eq 0 ]
    then
        echo "$1 结尾是.txt"
    else
        echo "$1 结尾不是.txt"
    fi
     
    [root@db03 scripts]# sh expr1.sh a.log
    a.log 结尾不是.txt
    [root@db03 scripts]# sh expr1.sh a.txt
    a.txt 结尾是.txt
     
    [root@db03 scripts]# read -t 5 -p "请输入一个数字:" a
    请输入一个数字:11
    [root@db03 scripts]# echo $a
    11
     

    1.9.6 选项操作

    [root@db03 scripts]# cat read1.sh 
    #!/bin/bash
    cat <<EOF
      1)install lamp
      2)install lnmp
      3)exit
    EOF
    read -t 15 -p "请输入一个数字:" a
    if [ $a -eq 1 ] 
    then
        echo "installing. lamp"
    elif [ $a -eq 2 ]
    then
        echo "installling lnmp"
    elif [ $a -eq 3 ]
    then
        echo "byebye"
        exit
    else
        echo "Usage $0 {1|2|3}"
        exit 1
    fi
     
    

    [root@db03 scripts]# cat chuancan.sh 
    #!/bin/bash
    #!/bin/bash
    a=$1
    b=$2
    if [ $# -ne 2 ]
    then
        echo "Usage:$0 num1 num2"
        exit 2
    fi

    expr $a + $b + 1 &>/dev/null
    if [ $? -ne 0 ]
    then
        echo "Usage:$0 num1 num2"
        exit 1
    fi


    echo "a-b=$(($a-$b))"
    echo "a+b=$(($a+$b))"
    echo "a*b=$(($a*$b))"
    echo "a/b=$(($a/$b))"
    echo "a**b=$(($a**$b))"
    echo "a%b=$(($a%$b))"  

     

     

     

     

    I am oldboy, I teach linux.

     

     

    for char in I am oldboy I teach linux

    do

        if [ ${#char} -lt 3 ]

        then

            echo $char

        fi

    done

    1.9.7 判断长度

  • 相关阅读:
    2014-写给明年现在的自己
    DDD 领域驱动设计-Value Object(值对象)如何使用 EF 进行正确映射
    DDD 领域驱动设计-在动手之前,先把你的脑袋清理干净
    醍醐灌顶:领域驱动设计实现之路
    拨乱反正:DDD 回归具体的业务场景,Domain Model 再再重新设计
    设计窘境:来自 Repository 的一丝线索,Domain Model 再重新设计
    No zuo no die:DDD 应对具体业务场景,Domain Model 重新设计
    拨开迷雾,找回自我:DDD 应对具体业务场景,Domain Model 到底如何设计?
    【记录】GitHub/TortoiseGit 修改邮箱/提交者
    死去活来,而不变质:Domain Model(领域模型) 和 EntityFramework 如何正确进行对象关系映射?
  • 原文地址:https://www.cnblogs.com/zdqc/p/9717083.html
Copyright © 2020-2023  润新知