shell变量及赋值
shell的变量
变量是用来临时保存数据的,并且该数据时可以变化的,任何一个语言都离不开变量,如果某个内容需要多次使用并且会重复出现,这样就可以使用变量了,如果需要修改直接修改变量就可以了
常见 Shell 变量的类型包括自定义变量、环境变量、只读变量、位置变量、预定义变量
环境变量
环境变量指的是出于运行需要而由 Linux 系统提前创建的一类变量,主要用于设置用户的工作环境,包括用户宿主目录、命令查找路径、用户当前目录、登录终端等。
环境变量的值由 Linux 系统自动维护,会随着用户状态的改变而改变。
使用 env 命令可以查看到当前工作环境下的环境变量,对于常见的一些环境变量应了解其各自的用途。例如,变量 USER 表示用户名称,HOME 表示用户的宿主目录,LANG 表示语言和字符集,PWD 表示当前所在的工作目录,PATH 表示命令搜索路径等、RANDOM表示随机数,会返回0-32767的整数,USER表示当前账户的账户名称等,一般都用全大写定义,注意和自定义变量区分
[root@localhost ~]# echo $RANDOM
20315
[root@localhost ~]# echo $HOME
/root
[root@localhost ~]# echo $USER
root
[root@localhost ~]# echo $UID
0
[root@localhost ~]# echo $PWD
/root
[root@localhost ~]# echo $LANG
zh_CN.UTF-8
[root@localhost ~]# echo $SHELL
/bin/bash
[root@client ~]# echo
R
A
N
D
O
M
/
/
l
i
n
u
x
中
RANDOM //linux中
RANDOM//linux中random用于生成0—32767的随机数
29780
PATH 变量用于设置可执行程序的默认搜索路径,当仅指定文件名称来执行命令程序时,Linux 系统将在 PATH 变量指定的目录范围查找对应的可执行文件,如果找不到则会提示“command not found”。
[root@localhost ~]# test.sh
bash: test.sh: 未找到命令…
[root@localhost ~]# echo $PATH
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin
[root@localhost ~]# pwd
/root
这时因为test.sh不在$PATH的目录里面,所以系统无法识别无法直接用,需要跟上绝对路径使用该脚本
方法一:将脚本的目录加入$PATH
[root@localhost ~]# PATH="$PATH:/root" //这时临时的,如果永久生效需要编辑/etc/profile文件
[root@localhost ~]# echo $PATH
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin:/root
[root@localhost ~]# test.sh
hello world
方法二:将你自己写的脚本放到$PATH中的某一个目录
在 Linux 系统中,环境变量的全局配置文件为/etc/profile,在此文件中定义的变量作用 于所有用户。除此之外,每个用户还有自己的独立配置文件(~/.bash_profile)。
修改完了要重新登陆才能生效,如果想立即生效,可以使用source
注意:修改$PATH需要慎重操作,如果找不到了会影响命令的使用!!!
例如:
[root@localhost ~]# PATH= //手误将PATH设为空
[root@localhost ~]# echo $PATH
[root@localhost ~]# ls
-bash: ls: 没有那个文件或目录
只读变量
Shell 变量中有一种特殊情况,一经设定,其值是不可改变的,这种变量被称为只读变 量。在创建变量的时候可将其设置为只读属性,也可以将已存在的变量设置为只读属性,只 读变量主要用于变量值不允许被修改的情况
只读变量不可以改变值也不可以被删除
[root@localhost ~]# test=123
[root@localhost ~]# readonly test #readonly用来定义只读变量,一旦使用readonly定义的变量在脚本中就不能更改
[root@localhost ~]# echo $test
123
[root@localhost ~]# test=456
-bash: test: 只读变量
[root@localhost ~]# unset test
-bash: unset: test: 无法反设定: 只读 variable
需要退出登陆
位置变量
当执行命令行操作时,第一个字段表示命令名或脚本程序名,其余的字符串参数按照从左到右的顺序依次赋值给位置变量。
位置变量也称为位置参数,使用$1、$2、$3、…、$9 表示
命令或脚本本身的名称使用“$0”表示
例:编写一个简单的创建用户和密码的脚本
[root@localhost ~]# vim user.sh
#!/bin/bash
useradd $1
echo $2 | passwd --stdin $1
[root@localhost ~]# bash user.sh sj 123
更改用户 sj 的密码 。
passwd:所有的身份验证令牌已经成功更新。
[root@localhost ~]#
[root@localhost ~]# id sj
uid=1001(sj) gid=1001(sj) 组=1001(sj)
列2
[root@localhost ~]# vim adder2num.sh
#!/bin/bash
SUM=expr $1 + $2
echo “$1 + $2 = $SUM”
[root@localhost ~]# chmod +x adder2num.sh
[root@localhost ~]# ./adder2num.sh 12 34 //$1 为 12、$2 为 34 的情况 12 + 34 = 46
[root@localhost ~]# ./adder2num.sh 56 78 //$1 为 56、$2 为 78 的情况 56 + 78 = 134
预定义变量
预定义变量是由 Bash 程序预先定义好的一类特殊变量,用户只能使用预定义变量,而 不能创建新的预定义变量,也不能直接为预定义变量赋值。预定义变量使用“$”符号和另一个符号组合表示
$#:表示命令行中位置参数的个数。
$*:表示所有位置参数的内容,这些内容当做一个整体
$@:表示列出所有位置参数,但是是以单个的形式的列出
$?:表示前一条命令执行后的返回状态,返回值为 0 表示执行正确,返回任何非 0 值均表示执行出现异常。
$0:表示当前执行的脚本或程序的名称
$$:表示返回当前进程的进程号
$!:返回最后一个后台进程的进程号
例1:根据一个简单的脚本来理解每个预定义和位置变量的含义
#!/bin/bash
echo $1
echo “
0
表
示
当
前
执
行
的
脚
本
或
程
序
的
名
称
”
e
c
h
o
“
0 表示当前执行的脚本或程序的名称” echo “
0表示当前执行的脚本或程序的名称”echo“# 表示命令行中位置参数的个数”
echo “
∗
所
有
位
置
参
数
的
内
容
,
这
些
内
容
当
做
一
个
整
体
”
e
c
h
o
“
* 所有位置参数的内容,这些内容当做一个整体” echo “
∗所有位置参数的内容,这些内容当做一个整体”echo“@ 表示列出所有位置参数,但是是以单个的形式的列出”
例2:可以将脚本做下优化
#!/bin/bash
echo “当前脚本名称为$0”
echo “当前脚本的第一个参数是$1”
echo “当前脚本的第二个参数是
2
"
e
c
h
o
"
当
前
脚
本
一
共
有
2" echo "当前脚本一共有
2"echo"当前脚本一共有#个参数”
……
例3 touch /home/kgc.txt &
echo ”$$ 表示返回当前进程的进程号“
echo “
?
0
正
确
,
1
"
e
c
h
o
"
? 0正确,1" echo "
?0正确,1"echo"! 返回最后一个后台进程的进程号"
理解
∗
和
*和
∗和@的区别
∗
、
*、
∗、@:表示命令或脚本要处理的参数。
$*:把所有参数看成以空格分隔的一个字符串整体(单字符串)返回,代表"$1 $2 $3 $4"。
$@:把各个参数加上双引号分隔成n份的参数列表,每个参数作为一个字符串返回,代表"$1"“$2"”$3"“$94”。
例1
区别 $* 和 $@
#!/bin/bash
echo “打印出$*”
for var in “
∗
"
d
o
e
c
h
o
"
*" do echo "
∗"doecho"var”
done
echo “打印出$@”
for var in “
@
"
d
o
e
c
h
o
"
@" do echo "
@"doecho"var”
done
在脚本中加入以下两行内容然后执行
touch “
∗
"
t
o
u
c
h
"
*" touch "
∗"touch"@”
ls -l查看创建了哪些文件
总结:
$*是将参数全部当做一个整体
$@是将参数每一个都当做单独的个体
set:查看系统所有的变量,包括环境变量和自定义变量(没有单独查看自定义变量的命令,可以set管道过滤)
自定义变量
1、变量的定义
Bash中的变量操作相对比较简单,不像其他高级编程语言(如C/C++、Java等)那么复杂。在定义一个新的变量时,一般不需要提前进行声明,而是直接指定变量名称并赋给初始值(内容)即可
格式:变量名=变量值
变量名:临时存放数据的地方
变量值:临时的可变化的数据
等号两边没有空格。变量名称需以字母或下划线开头,名称中不要包含特殊字符(如+、-、*、/、.、?、%、&、#等)
用echo查看和引用变量的值
通过在变量名称前添加前导符号“$”,可以引用一个变量的值,使用 echo 命令可以查看变量,可以在一条 echo 命令中同时查看多个变量值
列1
Product=Python
Version=2.7.13
echo P r o d u c t Product ProductVersion
当变量名称容易和紧跟其后的其他字符相混淆时,需要添加大括号“{}”将其括起来,否则将无法确定正确的变量名称。对于未定义的变量,将显示为空值
举例说明
列2
{}引用变量
echo ${Product}2.5
echo ${test}RMB
echo选项
echo -n 表示不换行输出
使用echo -e输出转义字符,将转义后的内容输出到屏幕上
常用的转义字符如下:
\c 不换行输出,在”\c”后面不存在字符的情况下,作用相当于echo -n
\n 换行
\t 转义后表示插入tab,即制表符
注:\转义符,跟在\之后的特殊符号将失去特殊含义,变为普通字符。如$将输出“$”符号,而不当做是变 量引用
例1
[root@localhost ~]# echo -n hello
hello[root@localhost ~]#
例2
[root@localhost ~]# echo -e “hello\t”
hello
例3 转义符案列
例:
echo $VAR1
echo $VAR1
$VAR1
取消定义
unset 变量名
特殊操作
还有一些特殊的赋值操作,可以更灵活地为变量赋值,以便适用于各种复杂的管理任务
双引号(”)
双引号主要起界定字符串的作用,特别是当要赋值的内容中包含空格时,必须以双引号括起来;其他情况下双引号通常可以省略
1、当内容中有空格
echo “hello world”
echo nihao
2、当以变量的值进行赋值
[root@localhost ~]# version=2
[root@localhost ~]# pyver=“python $version”
[root@localhost ~]# echo $pyver
python 2
单引号(‘)
当要赋值的内容中包含KaTeX parse error: Undefined control sequence: \等 at position 4: 、“、\̲等̲具有特殊含义的字符时,应使用单…test"
123
[root@localhost ~]# echo ‘$test’
$test
反撇号(`)
反撇号主要用于命令替换,允许将执行某个命令的屏幕输出结果赋值给变量。
反撇号括起来的范围内必须是能够执行的命令行,否则将会出错
ls -lh which useradd
先通过 which useradd 命令查找出 useradd 命令的程序位置,然后根据查找结果列出文件属性
date +%Y-%m-%d
[root@localhost ~]# time=date +%T
[root@localhost ~]# echo $time
04:23:22
使用反撇号难以在一行命令中实现嵌套命令替换操作,这时可以改用 “$()”来代替反撇号操作,以解决嵌套的问题
rpm -qc $(rpm -qf $(which useradd))
----交互式定义变量------
read 命令
除了上述赋值操作以外,还可以使用 Bash 的内置命令 read 来给变量赋值。
用来提示用户输入信息,从而实现简单的交互过程。执行时将从标准输入设备(键盘)读入 一行内容,并以空格为分隔符,将读入的各字段依次赋值给指定的变量(多余的内容赋值给 最后一个变量)。若指定的变量只有一个,则将整行内容赋值给此变量。
[root@localhost ~]# read test
123 //等待用户输入,把输入的值赋予test变量
[root@localhost ~]# echo $test
123
一般来说为了使交互式操作的界面更加友好,提高易用性,read 命令可以结合“-p”选项来设置提示信息,以便告知用户应该输入什么内容等相关事项
[root@localhost ~]# read -p “请输入你的姓名:” name
请输入你的姓名:humajun
[root@localhost ~]# echo $name
humajun
交互式定义变量(read)
-p 提示用户的信息
-n 定义字符数
-s 不显示用户输入的内容,常用于输入密码 read -s -p “input your password:” pass
-t 定义超时时间,超过多长时间没输自动退出
从文件读取内容赋值给变量
[root@server myscripts]# echo 192.168.100.100 > ip.txt
[root@server myscripts]# cat ip.txt
192.168.100.100
[root@server myscripts]# read -p “input your ip:” IP < ip.txt
[root@server myscripts]# echo $IP
192.168.100.100
stty -echo //关闭屏幕回显
stty echo //开启屏幕回显
–变量的作用范围—
默认情况下,新定义的变量只在当前的 Shell 环境中有效,因此称为局部变量,当进入子程序或新的子 Shell 环境时,局部变量将无法再使用
[root@localhost ~]# bash #进入子shell环境
[root@localhost ~]#
[root@localhost ~]# echo $name
[root@localhost ~]# echo $test
export命令
为了使用户定义的变量在所有的子 Shell 环境中能够继续使用,减少重复设置工作,可以通过内部命令 export 将指定的变量导出为全局变量。用户可以同时指定多个变量名称作为参数(无须使用“$”符号),变量名之间以空格分隔
[root@localhost ~]# exit
exit
[root@localhost ~]# export name test
[root@localhost ~]# bash
[root@localhost ~]# echo $name $testexit
humajun 123
或
[root@localhost ~]# echo “$Product KaTeX parse error: Expected 'EOF', got '#' at position 38: …ot@localhost ~]#̲ export Product…Product $Version” #子程序引用全局变量
Benet 6.0
[root@localhost ~]# exit
使用 export 导出全局变量的同时,也可以为变量进行赋值,这样在新定义全局变量时就不需要提前进行赋值了
env查看用户当前环境变量
export ABC=123
再次env就能看到了
export -n ABC 取消定义的全局变量变成局部变量