SHELL脚本-变量

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
上一篇:MFS分布式(一)


下一篇:seleinum 定位元素