经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 数据库/运维 » Linux/Shell » 查看文章
shell编程基础
来源:cnblogs  作者:Linux小菜鸟  时间:2024/7/11 11:10:16  对本文有异议

shell脚本编程基础

【1】、shell概述

  • shell是一门编程语言,如:C、C++、java、PHP、python Go等

  • 语言分类

    编程型语言:C、C++、Go等为编译型语言。程序运行需要提前编译,编译语言都有编译器

    解释型语言:shell、PHP、python等为解释型语言,程序在运行时不需要提前编译,一边执行,一边解释,每种解释型语言都有解释器

  • shell语言支持大部分编程语言都具备的功能:if判断、for循环、函数、数组、加减乘除、逻辑运算等

  • 脚本的本质就是命令的堆积

【2】、shell编写规范

系统解释器

  1. [root@xu script]# cat /etc/shells
  2. /bin/sh #Unix解释器
  3. /bin/bash #Linux默认解释器
  4. /usr/bin/sh
  5. /usr/bin/bash

这就是一个简单的shell script,在我们编写完成后,要给script加上可执行的权限

  1. #!/bin/bash
  2. #首先要进行环境指定,指定编译器,一般我们会使用/bin/bash,标准解释器
  3. # '#'表示注释、‘#!’表示指定解释器
  4. #功能:输出hello world
  5. echo hello world

在脚本中不能出现交互式命令

  1. #例如创建用户并且给用户设置密码
  2. #!/bin/bash
  3. #creat user
  4. useradd zhanghaowei
  5. echo 123 | passwd --stdin zhanghaowei
  6. #我们使用设置密码的方式采用非交互式的方式

【3】、如何写好一个shell脚本

  • 明确任务需求
  • 按照需求整理好每一个步骤,先后顺序
  • 运行脚本,并根据运行结果排除错误
  • 优化脚本并达到最后的结果
  • 脚本程序规范
    • 声明解释器(shebang)

      • !/bin/bash

    • 注释

      • 对脚本的描述
    • 执行指令

      • Linux命令

【4】、编写脚本

  • 自动配置本地yum仓库的脚本

    1. [BaseOS]
    2. name=local_BaseOS
    3. baseurl=file:///mnt/cdrom/BaseOS
    4. enable=1
    5. gpgcheck=0
    6. [AppStream]
    7. name=local_AppStream
    8. baseurl=file:///mnt/cdrom/AppStream
    9. enable=1
    10. gpgcheck=0
    11. EOF
    12. sleep 2
    13. echo "======永久挂载光驱======"
    14. echo "/dev/cdrom /mnt/cdrom iso9660 defaults 0 0" >> /etc/fstab
    15. sleep 2
    16. echo "======检查软件包数量======"
    17. yum list | wc -l

【5】、脚本的执行方式

  • 方式一:赋予脚本执行权限后,可用绝对路径和相对路径执行

  • 方式二:调用解释器执行脚本文件,会开启jie'shi'q

  • 方式三:source和 . :执行脚本文件

    1. #绝对路径执行脚本
    2. [root@xu script]# /script/hello.sh
    3. hello world
    4. #相对路径执行脚本
    5. [root@xu script]# ./hello.sh
    6. hello world
    1. #调用解释器执行脚本,调用解释器时可以不给执行权限
    2. [root@xu script]# bash /script/hello.sh
    3. hello world
  • 父子shell问题

    1. #我们知道脚本的执行是通过调用bash解释器去执行,我们通过sshd连接一个会话可以存在多个bash进程
    2. sshd(1065)───sshd(1896)───sshd(1900)───bash(1901)───bash(2093)───bash(2114)───pstree(2135)
    3. #通过pstree我们可以看到当前session存在多个bash环境,都是由systemd进程创建出来的,而我们现在处于bash(2114)环境中,如果此时我们执行脚本采用的环境时bash(2114)环境。
    4. #如果我们使用路径(source)去执行脚本和使用bash去执行脚本就会产生区别
    5. #采用.(source)去执行脚本,是在当前的bash环境下执行的,而使用bash或path去执行脚本则是在当前的bash环境下创建出一个子bash去执行脚本,并切换当前环境为新的脚本

【6】、变量

  • shell中变量类型只有string,没有别的;比方说在python中变量是有不同的类型的(list、dic、str。。。),但是在shell中只有str.

  • 变量的存在让脚本变得更加灵活,能够应对不同的场景

  • 语法:变量名="变量值"

  • $:调用变量

    1. [root@xu ~]# name=$(hostname)
    2. [root@xu ~]# echo $name
    3. xu

1、变量分类

(1)、系统定义变量

  1. # 系统自定义的系统变量
  2. [root@moudle01 17:23:53 /script]# echo $PWD
  3. /script
  4. [root@moudle01 17:23:59 /script]# echo $SHELL
  5. /bin/bash
  6. [root@moudle01 17:24:05 /script]# echo $HOME
  7. /root
  8. [root@moudle01 17:24:14 /script]# echo $USER
  9. root
  10. [root@moudle01 17:24:51 /script]# echo $UID
  11. 0

(2)、位置变量和预定义变量

在shell中也存在着和C语言相似的参数传递问题

只不过在shell中的参数问题是通过命令去完成的,我们在执行一个shell scrip时可以写上参数,将参数传递进shell script,同时在shell中去接收通过命令传入的参数。

位置参数:在执行脚本的命令中写上的参数就是位置参数,从1开始

在接收时可以使用以下标识符

  • $0:提取文件名(用source执行脚本时,不显示文件名而是解释器名字)
  • ${1}:提取位置参数为1的值
  • ...
  • ${20}: 提取位置参数为20的值
  • $*:接收所有参数
  • $@:接收所有参数
  • $$:提取当前进程的PId
  • $?:上一条命令的执行结果(上一个程序/命令的返回状态码)
  • $#:所有参数个数
  1. #!/bin/bash
  2. echo $1
  3. echo $2
  4. echo $3
  5. echo $0
  6. echo $*
  7. echo $#
  8. echo $$
  9. echo $?
  10. [root@moudle01 21:10:40 /script]# bash test04.sh a b c d
  11. a
  12. b
  13. c
  14. test04.sh
  15. a b c d
  16. 4
  17. 2728
  18. 0
  1. [root@moudle01 21:18:33 /script]# kjasdhfsdlgasgasgadfg
  2. -bash: kjasdhfsdlgasgasgadfg: command not found
  3. [root@moudle01 21:18:45 /script]# echo $?
  4. 127
  5. [root@moudle01 21:18:49 /script]# echo $?
  6. 0

创建用户tom,密码123

  1. #!/bin/bash
  2. useradd $1
  3. echo $2 | passwd --stdin $1

2、env和set

env:所有环境变量

set:所有变量

  1. [root@moudle01 21:30:10 /script]# abcd=6666666
  2. [root@moudle01 21:30:55 /script]# set | grep abcd
  3. abcd=6666666
  4. [root@moudle01 21:31:00 /script]# env | grep PATH
  5. PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin

3、常用的特殊符号

  • “”:双引号,引用整体,界定范围

  • ‘’:单引号,引用整体,并且取消所有特殊字符含义,界定范围

    1. [root@moudle01 21:32:13 /script]# a=" a"
    2. [root@moudle01 21:36:17 /script]# echo "$a"
    3. a
    4. # 如果有特殊符号,定义时单引号和双引号都行,在引用时只能使用双引号
    5. [root@moudle01 21:36:25 /script]# echo '$a'
    6. $a
  • $[]、$(())、expr:四则运算

    1. [root@xu script]# echo $[1+1]
    2. 2
    3. [root@xu script]# echo $[2*3]
    4. 6
    5. [root@xu script]# echo $[2-3]
    6. -1
    7. [root@xu script]# echo $[6/3]
    8. 2
    9. [root@xu script]# echo $[10%3]
    10. 1
    11. # expr 运算符号两侧必须加空格
    12. [root@moudle01 16:13:56 ~]# expr 1 + 1
    13. 2
  • $():取命令结果作为参数

  • ``:反撇号,将命令的输出结果作为参数

    1. [root@moudle01 21:40:57 /script]# date
    2. Wed Jun 12 21:41:07 CST 2024
    3. [root@moudle01 21:38:26 /script]# a=`date`
    4. [root@moudle01 21:40:54 /script]# echo $a
    5. Wed Jun 12 21:40:54 CST 2024
    6. [root@moudle01 21:41:07 /script]# x=$(ls)
    7. [root@moudle01 21:42:37 /script]# echo $x
    8. abc rsync_inotify.sh test01.sh test02.sh test03.sh test04.sh test05.sh

4、read

交互式输入变量

  1. [root@moudle01 21:48:04 /script]# read -p "请输入:" a
  2. 请输入:123
  3. [root@moudle01 21:48:47 /script]# echo $a
  4. 123
  1. #!/bin/bash
  2. read -p "请输入要创建的用户名:" a
  3. useradd $a
  4. read -p "请输入用户的密码:" p
  5. echo $p | passwd --stdin $a
  6. [root@moudle01 21:50:41 /script]# source test05.sh
  7. 请输入要创建的用户名:xxx
  8. 请输入用户的密码:123
  9. Changing password for user xxx.
  10. passwd: all authentication tokens updated successfully

5、屏蔽回显

在敲完之后,你在后面输入的所有内容都看不到

  1. [root@moudle01 21:51:16 /script]# stty -echo

显示回显

  1. [root@moudle01 21:51:16 /script]# stty echo

写入创建用户的脚本,让密码输入更加安全

  1. #!/bin/bash
  2. read -p "请输入要创建的用户名:" a
  3. useradd $a
  4. stty -echo
  5. read -p "请输入用户的密码:" p
  6. stty echo
  7. echo ""
  8. echo $p | passwd --stdin $a
  9. [root@moudle01 21:59:24 /script]# source test05.sh
  10. 请输入要创建的用户名:aaa
  11. 请输入用户的密码:
  12. Changing password for user aaa.
  13. passwd: all authentication tokens updated successfully.

6、export命令

  • 环境变量(全局变量):系统在全局生效的变量

    普通变量(局部变量):只针对局部环境生效,如当前某一个login.bash脚本生效

    全局变量和局部变量就是相对于父子shell(bash)。环境变量(全集变量)就是父shell也可以使用,普通变量(局部变量)只有子shell可以使用

  • 变量生存周期

    1、永久变量:写入文件,反复读取、加载,让其永久生效,如/etc/profile的PATH修改

    2、临时变量:如命令行export定义的一个变量存储数据,关闭shell后失效

  • 在定义变量时有两种选择

  • 1、加export:在父shell中定义的变量可以在子shell中去使用,也就是全局变量

      1. [root@xu ~]# export name="xuruizhao"
      2. [root@xu ~]# echo ${name}
      3. xuruizhao
      4. [root@xu ~]# bash
      5. Skill is acquired through repeated practice, and practice makes perfect.
      6. [root@xu ~]# bash
      7. Skill is acquired through repeated practice, and practice makes perfect.
      8. [root@xu ~]# echo ${name}
      9. xuruizhao

    2、不加export:只对当前的shell有效,子shell看不到,也就是局部变量

      1. [root@xu ~]# name="hello world"
      2. [root@xu ~]# echo ${name}
      3. hello world
      4. [root@xu ~]# bash
      5. #我们切换了bash环境,也就是进入了子shell,由于没有加export子shell中是不能使用父shell中定义的变量
      6. Skill is acquired through repeated practice, and practice makes perfect.
      7. [root@xu ~]# echo ${name}
      8. [root@xu ~]#

7、let定义变量

使用let定义变量,可以在定义时进行计算

  1. [root@moudle01 22:13:42 /script]# let a=1+1
  2. [root@moudle01 22:14:21 /script]# echo $a
  3. 2
  4. [root@moudle01 22:16:30 /script]# let a+=3
  5. [root@moudle01 22:16:35 /script]# echo $a
  6. 5

8、bc命令

上面的运算均不支持小数

bc命令支持小数

  1. [root@moudle01 16:13:58 ~]# echo "1.1+1" | bc
  2. 2.1

9、变量定义初值

  1. #!/bin/bash
  2. read -p "请输入用户名" username
  3. if [ -z $username ];then
  4. echo "用户名不能为空" && exit
  5. fi
  6. useradd $username
  7. read -p "请输入密码" pass
  8. echo ${pass:-123456} | passwd --stdin $username
  9. [root@moudle01 11:37:41 /script]# bash user_add02.sh
  10. 请输入用户名
  11. 用户名不能为空

【7】、变量测试

两种测试格式

test

[]

1、字符串

(1)、数组测试

== 判断是否相同

! = 判断两侧是否不等

  1. [root@moudle01 16:29:46 ~]# a=1
  2. [root@moudle01 16:29:51 ~]# b=2
  3. [root@moudle01 16:29:52 ~]# [ $a != $b ]
  4. [root@moudle01 16:30:04 ~]# echo $?
  5. 0

如果一个变量值为空,我们在进行比较时,

1、可以加上双引号

2、-z:判断变量是否为空

? ! -Z/-n:判断变量是否非空

  1. [root@moudle01 16:33:11 ~]# [ a == "$c" ]
  2. [root@moudle01 16:33:23 ~]# echo $?
  3. 1
  4. [root@moudle01 16:33:27 ~]# [ -z "$c" ]
  5. [root@moudle01 16:35:13 ~]# echo $?
  6. 0
  7. [root@moudle01 16:35:16 ~]# [ -z "$a" ]
  8. [root@moudle01 16:35:22 ~]# echo $?
  9. 1
  10. [root@moudle01 16:38:16 ~]# c=123
  11. [root@moudle01 16:38:23 ~]# [ -n "$c" ]
  12. [root@moudle01 16:38:26 ~]# echo $?
  13. 0

(2)、逻辑测试

&&与

||或

  1. #!/bin/bash
  2. [ $USER == root ] || echo "非管理员禁止执行该脚本" && exit
  3. yum -y install vsftpd > /dev/null
  4. systemctl start vsftpd
  5. systemctl enable vsftpd
  6. # 只有root用户可以执行此脚本
  7. #!/bin/bash
  8. [ $USER != root ] && echo "非管理员禁止执行" && exit2
  9. yum -y install vsftpd > /dev/null
  10. systemctl start vsftpd
  11. systemctl enable vsftpd
  1. [root@moudle01 16:57:25 /script]# ls a || ls b && ls c || ls && ls /etc/yum.repos.d/
  2. a
  3. c
  4. bak epel local_yum_house.repo

2、数字

-eq:相等

-ne:不相等

-gt:大于

-ge:小于等于

-lt:小于

-le:小于等于

  1. #!/bin/bash
  2. [ $# -ne 2 ] && echo "输入变量数量错误" && exit
  3. useradd $1
  4. echo $2 | passwd --stdin $1

3、文件

-e:判断文件是否存在,不关心文件类型

-f:判断文件是否存在,必须是普通文件

-d:判断文件是否存在,必须是目录

  1. [root@moudle01 17:19:49 /script]# [ -e a ]
  2. [root@moudle01 17:20:00 /script]# echo $?
  3. 0
  4. [root@moudle01 17:20:03 /script]# [ -d a ]
  5. [root@moudle01 17:20:09 /script]# echo $?
  6. 1
  7. [root@moudle01 17:20:10 /script]# [ -f x ]
  8. [root@moudle01 17:20:21 /script]# echo $?
  9. 1

-r:判断当前用户对文件是否有读权限(对root无效,只针对普通用户)

-w:判断当前用户对文件是否有写权限(对root无效,只针对普通用户)

-x:判断当前用户对文件是否有x权限

  1. [root@moudle01 17:21:45 /script]# [ -r a ]
  2. [root@moudle01 17:22:02 /script]# echo $?
  3. 0
  4. [root@moudle01 17:22:05 /script]# [ -w a ]
  5. [root@moudle01 17:22:09 /script]# echo $?
  6. 0
  7. [root@moudle01 17:22:10 /script]# [ -x a ]
  8. [root@moudle01 17:22:17 /script]# echo $?
  9. 1

4、编写脚本

每隔两分钟检查服务器的账户数量,如果发现增加,则给管理员发邮件

  1. #!/bin/bash
  2. num=$(cat /etc/passwd | wc -l)
  3. [ num -gt 38 ] && echo "用户增加" | mail -s "user_info" root
  4. # 编写计划任务
  5. */2 * * * * /script/test02.sh

原文链接:https://www.cnblogs.com/xuruizhao/p/18295168

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号