SHELL脚本-变量
SHELL脚本介绍
编程语言分类
编译型语言
- 程序执行前需要编译成机器语言,直接运行编译结果。程序执行效率高,依赖编译器,跨平台性较差。如C、C++等。适应于底层开发或者大型应用程序或者操作系统开发。
解释性语言
- 程序不需要编译,运行时由解释器翻译成机器语言,每执行一次翻译一次,效率较低,跨平台性较好。如Python/JavaScript/ Perl /ruby/Shell等。适用于一些服务器脚本及一些辅助的接口,对速度要求不高、对各个平台的兼容性有要求的项目。
shell概述
-
shell是人机交互的一个桥梁,用于解析命令,交给系统内核,从而实现物理硬件的管理
-
常见shell
[root@server1 ~]# cat /etc//shells
/bin/sh 是bash的快捷方式
/bin/bash Linux默认shell,几乎包含shell所有的功能
/usr/bin/sh
/usr/bin/bash
/sbin/nologin 非交互式shell,不能登录操作系统
shell脚本
-
将要执行的命令按一定的格式,一定的语法保存至脚本中,按顺序执行,进而实现特定的功能
-
适用场景:重复化,复杂化的工作
- 自动化备份
- 批量部署安装
- 自动化分析处理
-
脚本基本要求
脚本命名:建议以".sh"结尾
[root@server1 ~]# vim test.sh
1 #!/bin/bash <--脚本第一行,魔法字符,用于指定脚本bash,而非注释信息
2
3 # Name:test.sh <--注释,描述脚本的基本信息,建议写明,方便阅读
4 # Desc:打印'hello linux'
5 # Path:/root
6 # Usage:./test.sh
7 # Update:2021-15-07
8
9 #commands <--具体命令,实现特定功能
10 echo 'hello linux'
11 date +%F
脚本执行方法
标准执行
推荐,常用方法
给脚本加上执行权限
[root@server1 ~]# chmod +x test.sh
[root@server1 ~]# ls
anaconda-ks.cfg test.sh(绿色)
执行脚本
[root@server1 ~]# /root/test.sh <--绝对路径
hello linux
2021-09-05
[root@server1 ~]# ./test.sh <--相对路径
hello linux
2021-09-05
使用bash执行脚本
不要求脚本具有执行权限,常用此方法进行排错
[root@server1 ~]# bash test.sh
hello linux
2021-09-05
[root@server1 ~]# bash -x test.sh <--显示执行过程
+ echo 'hello linux'
hello linux
+ date +%F
2021-09-05
[root@server1 ~]# bash -n test.sh <--检查shell脚本的语法,无输出表示语法无误
其他方法执行脚本
仅用于编写配置文件的情况
[root@server1 ~]# source test.sh
hello linux
2021-09-05
[root@server1 ~]# . test.sh
hello linux
2021-09-05
变量
分类
本地变量(临时)
当前用户自定义的变量,只针对当前进程中有效,对于其他进程及当前进程的子进程无效。
[root@server1 ~]# a=hello
[root@server1 ~]# echo $a
hello
[root@server1 ~]# ps
PID TTY TIME CMD
8481 pts/0 00:00:00 bash
8585 pts/0 00:00:00 ps
[root@server1 ~]# /bin/bash
[root@server1 ~]# ps
PID TTY TIME CMD
8481 pts/0 00:00:00 bash #bash
8601 pts/0 00:00:00 bash #bash子进程
8614 pts/0 00:00:00 ps
[root@server1 ~]# echo $a
本地变量无效
[root@server1 ~]#
环境变量
当前进程有效,且能够被子进程调用
[root@server1 ~]# A=hello
[root@server1 ~]# echo $A
hello
env查看环境变量
[root@server1 ~]# env |grep '^A'
临时将本地变量加入到环境变量中
[root@server1 ~]# export A
[root@server1 ~]# env |grep '^A'
A=hello
永久加入环境变量
[root@server1 ~]# echo 'export A=hello' >> /etc/profile
[root@server1 ~]# tail -1 /etc/profile
export A=hello
全局变量
所有的用户和程序都能调用
重要文件
- ~/.bashrc:存放当前用户的bash信息,如用户定义的别名,umask值,函数等
- ~/.bash_profile:存放当前用户的环境变量
- /etc/bashrc :存放全局的bash信息
- /etc/profile:存放全局的环境变量
/etc/profile–> ~/.bash_profile–> ~/.bashrc -->/etc/bashrc -->~/.bash_logout(退出shell前读取的最后文件)
系统变量
bash内置变量,具有固定含义
$? 上一条命令执行的结果
0,正常退出
127,未找到命令
126,权限不够
1&2,表示没有那个文件或目录
[root@server1 ~]# ls
anaconda-ks.cfg
[root@server1 ~]# echo $?
0
[root@server1 ~]# ifconfig
-bash: ifconfig: 未找到命令
[root@server1 ~]# echo $?
127
[root@server1 ~]# vim test.sh
[root@server1 ~]# ./test.sh
-bash: ./test.sh: 权限不够
[root@server1 ~]# echo $?
126
[root@server1 ~]# ls aaa
ls: 无法访问aaa: 没有那个文件或目录
[root@server1 ~]# echo $?
2
$$ 当前进程的pid
[root@server1 ~]# ps
PID TTY TIME CMD
8478 pts/0 00:00:00 bash
11599 pts/0 00:00:00 ps
[root@server1 ~]# echo $$
8478
[root@server1 ~]# kill -9 `echo $$`
Connection closing...Socket close.
Connection closed by foreign host.
Disconnected from remote host(192.168.226.10:22) at 21:22:38.
Type `help' to learn how to use Xshell prompt.
[D:\~]$
$! 后台运行的最后那个进程的pid
[root@server1 ~]# sleep 1000 &
[1] 11631
[root@server1 ~]# sleep 2000 &
[2] 11632
[root@server1 ~]# jobs
[1]- 运行中 sleep 1000 &
[2]+ 运行中 sleep 2000 &
[root@server1 ~]# echo $!
11632
!$ 调用最后一条命令历史中的最后一个参数,如果没有参数,就调用命令本身
[root@server1 ~]# vim /etc/sysconfig/network-scripts/ifcfg-ens33
[root@server1 ~]# cat !$
cat /etc/sysconfig/network-scripts/ifcfg-ens33
TYPE="Ethernet"
PROXY_METHOD="none"
BROWSER_ONLY="no"
BOOTPROTO="static"
DEFROUTE="yes"
IPV4_FAILURE_FATAL="no"
IPV6INIT="yes"
IPV6_AUTOCONF="yes"
IPV6_DEFROUTE="yes"
IPV6_FAILURE_FATAL="no"
IPV6_ADDR_GEN_MODE="stable-privacy"
NAME="ens33"
UUID="79065771-0af6-41b9-9112-1dd5b373f8ed"
DEVICE="ens33"
ONBOOT="yes"
IPADDR=192.168.226.10
GATEWAY=192.168.226.2
NETMASK=255.255.255.0
DNS1=114.114.114.114
!! 调用最后一条历史命令
[root@server1 ~]# !!
cat /etc/sysconfig/network-scripts/ifcfg-ens33
TYPE="Ethernet"
PROXY_METHOD="none"
BROWSER_ONLY="no"
BOOTPROTO="static"
DEFROUTE="yes"
IPV4_FAILURE_FATAL="no"
IPV6INIT="yes"
IPV6_AUTOCONF="yes"
IPV6_DEFROUTE="yes"
IPV6_FAILURE_FATAL="no"
IPV6_ADDR_GEN_MODE="stable-privacy"
NAME="ens33"
UUID="79065771-0af6-41b9-9112-1dd5b373f8ed"
DEVICE="ens33"
ONBOOT="yes"
IPADDR=192.168.226.10
GATEWAY=192.168.226.2
NETMASK=255.255.255.0
DNS1=114.114.114.114
$# 脚本接入参数的个数
$* 脚本后面的所有参数,参数间用空格隔开,所有参数为一个整体
$@ 脚本后面的所有参数,参数全部输出,参数彼此独立
$0 当前程序
$1-9 位置参数变量,$1表示传入脚本的第一个参数
${10}-{n} 位置参数变量,{}为了界定范围
[root@server1 ~]# vim 1.sh
1 #!/bin/bash
2
3 #Desc=测试系统变量
4
5 echo "\$# = $#"
6 echo "\$* = $*"
7 echo "\$@ = $@"
8 echo "\$0 = $0"
9 echo "\$1 = $1"
10 echo "\$2 = $2"
11 echo "\$3 = $3"
12 echo "\$10 = ${10}"
[root@server1 ~]# chmod +x 1.sh
[root@server1 ~]# ./1.sh
$# = 0
$* =
$@ =
$0 = ./1.sh
$1 =
$2 =
$3 =
$10 =
[root@server1 ~]# ./1.sh a b c
$# = 3
$* = a b c
$@ = a b c
$0 = ./1.sh
$1 = a
$2 = b
$3 = c
$10 =
[root@server1 ~]# ./1.sh 1 2 3 4 5 6 7 8 9 10
$# = 10
$* = 1 2 3 4 5 6 7 8 9 10
$@ = 1 2 3 4 5 6 7 8 9 10
$0 = ./1.sh
$1 = 1
$2 = 2
$3 = 3
$10 = 10
系统变量 | 含义 | 备注 |
---|---|---|
$? | 上一条命令执行的结果 | 0,正常退出 |
$$ | 当前进程的pid | |
$! | 后台运行的最后那个进程的pid | |
!$ | 调用最后一条命令历史中的最后一个参数 | 没有参数,就调用命令本身 |
!! | 调用最后一条历史命令 | |
$# | 脚本接入参数的个数 | |
$* | 脚本后面的所有参数 | 所有参数为一个整体 |
$@ | 脚本后面的所有参数 | 参数彼此独立 |
$0 | 当前程序 | |
$1-9 | 位置参数变量 | $1表示传入脚本的第一个参数 |
${10}-{n} | 位置参数变量 | {}为了界定范围 |
定义变量
适用场景:
- 某个内容多次使用,且在代码中重复出现
- 将命令执行的结果定义为变量,后续想使用命令结果,直接调用变量
定义变量规则
变量名=变量值
等号两边不能有空格
区分大小写
变量名不能以数字或者特殊字符开头
变量值更改时,会覆盖
[root@server1 ~]# A=hello
[root@server1 ~]# B='hello linux'
[root@server1 ~]# C=`date +%F`
获取变量
[root@server1 ~]# echo $A
hello
[root@server1 ~]# echo ${#A} #获取变量长度
5
[root@server1 ~]# echo ${B}
hello linux
[root@server1 ~]# echo ${B:0:5}
hello
[root@server1 ~]# echo ${B:6:11}
linux
[root@server1 ~]# echo $C
2021-08-21
删除变量
[root@server1 ~]# unset A
[root@server1 ~]# echo $A
[root@server1 ~]#
declare有类型变量
选项说明
-i 将变量看成整数
-r readonly变量只读,不能删除,不能覆盖
-x 导出为环境环境
-a 指定为索引数组(普通数组);查看普通数组
-A 指定为关联数组;查看关联数组
[root@server1 ~]# declare -x D=hi
[root@server1 ~]# env |grep '^D'
D=hi
read交互
选项说明:
-p 提示信息
-n 指定变量值的最大字符数
-s 不显示
-t 指定超时时间
[root@server1 ~]# read -p "请输入你的名字:" name
请输入你的名字:jack
[root@server1 ~]# echo $name
jack
[root@server1 ~]# read -s -p "请输入你的密码:" passwd
请输入你的密码:[root@server1 ~]#
[root@server1 ~]# echo $passwd
1
[root@server1 ~]# read A B
aaa bbb
[root@server1 ~]# echo $A
aaa
[root@server1 ~]# echo $B
bbb
[root@server1 ~]# cat test.txt
user01 123456
[root@server1 ~]# read user password < test.txt
[root@server1 ~]# echo $user
user01
[root@server1 ~]# echo $password
123456
dirname&basename
dirname获取变量路径名
basename获取变量文件名
[root@server1 ~]# A=/etc/sysconfig/network-scripts/ifcfg-ens33
[root@server1 ~]# dirname $A
/etc/sysconfig/network-scripts
[root@server1 ~]# basename $A
ifcfg-ens33
数组
- 普通数组:只能使用数字作为数组索引
- 关联数组:可以使用字符串作为数组索引
定义数组
[root@server1 ~]# array[0]=v1
[root@server1 ~]# array[1]=v2
[root@server1 ~]# array[3]=v3
[root@server1 ~]# list=(v1 v2 v3 v4 v5 v6)
获取数组
[root@server1 ~]# echo ${array[*]}
v1 v2 v3
[root@server1 ~]# echo ${list[*]}
v1 v2 v3 v4 v5 v6
查看数组索引
[root@server1 ~]# declare -a |grep list
declare -a list='([0]="v1" [1]="v2" [2]="v3" [3]="v4" [4]="v5" [5]="v6")'
[root@server1 ~]# echo ${!list[@]}
0 1 2 3 4 5 6
切片
[root@server1 ~]# echo ${list[*]} #获取所有
v1 v2 v3 v4 v5 v6
[root@server1 ~]# echo ${list[@]} #获取所有,同上
v1 v2 v3 v4 v5 v6
[root@server1 ~]# echo ${list[1]} #按索引获取
v2
[root@server1 ~]# echo ${list[*]:1:3} #获取部分
v2 v3 v4
[root@server1 ~]# list[5]=value6 #修改
[root@server1 ~]# echo ${list[*]}
v1 v2 v3 v4 v5 value6
[root@server1 ~]# echo ${list[*]} #追加
v1 v2 v3 v4 v5 value6 v7
[root@server1 ~]# echo ${#list[*]} #获取元素个数
7
-------------------------------------------------------------------
定义关联数组
[root@server1 ~]# declare -A asso_array #声明其为关联数组
[root@server1 ~]# asso_array[one]=v1
[root@server1 ~]# asso_array[two]=v2
[root@server1 ~]# asso_array[three]=v3
[root@server1 ~]# declare -A digi_array
[root@server1 ~]# digi_array=([aaa]=1 [bbb]=2 [ccc]=3)
获取关联数组
[root@server1 ~]# echo ${asso_array[*]} #获取全部
v2 v3 v1 #关联数组是无序的
[root@server1 ~]# echo ${digi_array[*]}
2 3 1
[root@server1 ~]# echo ${asso_array[one]} #按索引获取
v1
[root@server1 ~]# echo ${#asso_array[@]} #获取元素个数
3
查看数组索引
[root@server1 ~]# declare -A |grep digi_array
declare -A digi_array='([bbb]="2" [ccc]="3" [aaa]="1" )'
[root@server1 ~]# echo ${!digi_array[@]}
bbb ccc aaa