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
<think>好的,用户现在问的是关于shell脚本中的 `-z` 参数。之前他们遇到了容器相关的错误,现在转向了shell脚本的问题,可能是在自动化或运维中遇到需要检查变量或字符串的情况。首先,我需要确认用户对shell脚本的基础了解程度,但根据之前的提问,他们可能有一定的技术背景。 `-z` 是用于判断字符串是否为空的条件测试。用户可能在使用if语句时遇到了问题,或者想知道如何正确使用这个参数。需要详细解释它的用法,包括语法、常见场景和示例。同时,用户可能有实际的问题需要解决,比如脚本变量未设置导致错误,使用`-z`来预防。 要确保回答结构清晰,分点说明,结合例子,这样用户更容易理解。可能还需要对比其他类似参数,比如`-n`,帮助用户全面了解。另外,注意用户之前的问题涉及安全机制,可能他们编写脚本时需要考虑健壮性,所以强调`-z`在预防错误中的作用很重要。 需要检查是否有其他相关参数或常见错误需要提及,比如变量未加引号导致的问题,或者在测试时的语法细节(如空格)。还要提醒用户注意shell的不同版本可能会有差异,但`-z`在大多数情况下是通用的。 最后,总结使用场景和注意事项,确保用户能正确应用`-z`,避免常见陷阱。如果有必要的话,可以给出实际脚本的例子,展示如何在具体情境中使用这个测试条件。</think>--- ### **Shell 脚本中的 `-z` 参数详解** 在 Shell 脚本中,`-z` 是 **条件测试表达式** 中的一个操作符,用于判断字符串是否为空。它是编写脚本时处理变量、输入校验和逻辑控制的关键工具。 --- ### **1. 语法与功能** - **语法**:`if [ -z "$VAR" ]` 其中 `$VAR` 是要检查的变量或字符串。 - **功能**: 如果 `$VAR` **未定义**或值为**空字符串**,则条件为真(返回退出状态码 `0`);否则为假(返回非零状态码)。 --- ### **2. 典型使用场景** #### **场景 1:检查变量是否为空** ```bash #!/bin/bash name="" # 赋值为空 if [ -z "$name" ]; then echo "变量 name 为空或未定义" else echo "变量 name 的值为: $name" fi ``` **输出**:`变量 name 为空或未定义` --- #### **场景 2:校验脚本输入参数** ```bash #!/bin/bash if [ -z "$1" ]; then echo "错误:未提供命令行参数" exit 1 fi echo "第一个参数是: $1" ``` **运行**: - `./script.sh` → 输出错误并退出 - `./script.sh hello` → 输出 `第一个参数是: hello` --- #### **场景 3:依赖环境变量检查** ```bash #!/bin/bash if [ -z "$JAVA_HOME" ]; then echo "JAVA_HOME 未配置,请先设置环境变量" exit 1 fi echo "Java 路径: $JAVA_HOME" ``` --- ### **3. 常见注意事项** - **变量引号包裹**: 必须将变量用双引号包裹(如 `"$VAR"`),否则未定义变量会导致语法错误。例如: ```bash [ -z $VAR ] # 若 VAR 未定义,等价于 [ -z ],语法错误 [ -z "$VAR" ] # 正确写法 ``` - **与 `-n` 的对比**: `-n` 是 `-z` 的反向操作符,用于判断字符串**非空**: ```bash if [ -n "$VAR" ]; then echo "VAR 非空" fi ``` - **避免过度检查**: 若变量可能包含空格或特殊字符(如 `VAR=" "`,一个空格),`-z` 会将其判定为**非空**。此时需根据业务逻辑进一步处理。 --- ### **4. 底层原理** - `-z` 是 Shell 内置的条件测试操作符,属于 `test` 命令(或 `[ ]` 语法)的一部分。 -Shell 解析时,`[ -z "$VAR" ]` 会被展开为对 `$VAR` 长度的判断。若长度为 0,则条件成立。 --- ### **5. 扩展用法** #### **组合其他条件** ```bash # 检查变量是否为空,且文件是否存在 if [ -z "$FILE" ] || [ ! -f "$FILE" ]; then echo "文件未指定或不存在" fi ``` #### **替代写法** ```bash # 使用双中括号(更灵活,支持正则匹配) if [[ -z $VAR ]]; then echo "VAR 为空" fi ``` --- ### **总结** `-z` 是 Shell 脚本中用于**空值校验**的核心操作符,常用于: 1. 验证输入参数 2. 检查环境变量 3. 避免未定义变量导致的错误 合理使用可提升脚本的健壮性,但需注意变量引号和边界条件处理。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值