Shell_02_变量
一、什么是变量
简单说就是让一个特定的字符串代表不固定的内容
yy=123
yy是变量的名字,123是变量的值
echo $yy //查看变量的值
就是用一个简单的好记的字符串,来取代比较复杂或者容易变动的数据
二、变量的设定
设定规则:
- 变量名与变量内容以一个等号
=
的连结,且等号两边不能有空格。如下所示:
myname=XiguaTian
- 变量名称只能是英文字母与数字,但是开头字符不能是数字,如下为错误:
2myname=xiguatian
- 变量内容若有空格符可使用双引号
\"
或单引号
\'
将变量内容组合起来,但
• 双引号内的特殊字符如
<article class=\"_2rhmJa\" 、
!` 等,可以保有原本的特性,如下所示:
var=\"lang is $LANG\"#则echo $var#可得lang is en_US
感叹号
!
不可以在shell命令行里直接在双引号里使用,但是可以在脚本中使用。
• 单引号内的特殊字符则仅为一般字符 (纯文本),如下所示:
var=\'lang is $LANG\'# 则echo $var#可得lang is $LANG
- 可用转意符
\\
将特殊符号(如
[Enter]
,
<article class=\"_2rhmJa\",``,
空格符`, 等)变成一般字符;
- 在一串命令中,还需要藉由其他的命令提供的信息,可以使用反单引号
命令
或
$(命 令)
(推荐这种)。特别注意,那个反引号
是键盘上方癿数字键
1` 左边那个按键,而不是单引号!
例如想要取得
核心版本的内容:
version=$(uname -r)# 再echo $version#可得4.9.125-linuxkit
- 增加变量的内容时,则可用
\"$变量名称\"字符串
或
${变量}字符串
(推荐)累加内容,如下所示:
PATH=\"$PATH\":/home/binPATH=${PATH}:/home/bin
- 若该变量需要在其他子程序执行,则需要以
export
来使变量变成环境变量:
export PATH
- 通常大写字符为系统默认变量,自定义变量可以使用小写字符,方便判断 (纯粹个人习惯) ;
- 取消变量的方法为使用 unset :
unset 变量名称
例如取消
myname
的设定:
unset myname
三、 环境变量
这种变量是会影响bash环境操作的,会在真正进入系统前由一个bash程序读入到系统中。通常都环境变量的名字以大写字符命名。
常见环境变量
-
PATH HOME MAIL SHELL PWD USERNAME UID ID
等。
-
RANDOM 随机数
echo $RANDOM
-
PS1:(提示字符的设置)
\\d :可显示出“星期 月 日”的日期格式,如:“Mon Feb 2”
\\H :完整的主机名称。如www.sharkyun.com\\h :仅取主机名称在第一个小数点之前的名字,
www\\t :显示时间,为 24 小时格式的“HH:MM:SS”
\\T :显示时间,为 12 小时格式的“HH:MM:SS”
\\A :显示时间,为 24 小时格式的“HH:MM”
@ :显示时间,为 12 小时格式的“am/pm”样式
\\u :目前使用者的帐号名称,如“dmtsai”;
\\v :BASH 的版本信息
\\w :完整的工作目录名称,由根目录写起的目录名称。但主文件夹会以 ~ 取代;
\\W :利用 basename 函数取得工作目录名称,所以仅会列出最后一个目录名。:下达的第几个指令。
读取环境变量的值
echo $HOME
列出shell环境下的所有环境变量及其内容
-
env
//
env
是 environment (环境) 的简写,默认的环境变量
-
set
// 列出系统中所有的变量,包括自定义的变量
-
export
变量名 // 使自定义的变量 成为 环境变量,环境变量可以被继承
子程序仅会继承父程序的环境变量, 子程序不会继承父程序的自订变量
name=sharkexport namebashecho $name
bash 的环境变量文件
- longin shell
取得shell时需要完整的登入流程;特点是登入时需要用户帐号和密码 - non-login shell
取得shell时不需要再次输入帐号和密码的情况下,所得到的shell
longin shell 会读取以下两个文件:
- ·/etc/profile· :这是系统整体设定,最好不要修改
- ·/.bash_profile
或
/.bash_login
或
~/.profile` :属于个人的配置文件
/etc/profile
会主动依序调用以下脚本文件:
-
/etc/inputrc
:定义快捷键
-
/etc/profile.d/*sh
:定义bash操作接口颜色、语系、命令别名等
-
etc/locale.conf
:定义系统的默认语系
bash 在读完
/etc/profile
后,接下来会读取以下3个文件,且只会读去一个,会按照以下顺序优先读取
~/.bash_profile
//会调用
~/.bashrc
,
也会有新的环境变量在下面的文件中被添加
~/.bash_login
~/.profile
最终,
~/.bashrc
才是最后被读入到系统环境中的文件
让这些环境变量文件中的变量等设置及时在当前
shell
终端中生效,有下两种方式
source ~/.bashrc或者. ~/.bashrc
no-longin shell
当取得
no-longin shell
时,该
shell
仅会读取
~/.bashrc
文件 而
~/.bashrc
最后又会调用
/etc/bashrc
/etc/bashrc
的作用:
• 依据不同的UID定义出
umask
值
• 依据不同的UID定义出提示符(就是PS1变量)
• 呼叫
/etc/profile.d/*.sh
的设定
其他的相关配置文件
•
/etc/man.config
这个文件最重要的就是定义了
MANPATH
这个变量,它定义了man page 的路径;在以tarball的方式安装软件时有用
•
~/.bash_history
历史命令记录文件;记录的数量与HISTFILESIZE变量有关。在
/etc/profile
里
•
~/.bash_logout
记录了当我注销bash后,系统再帮我做完什么动作后才离开的。
四、预定义变量
预定义的特殊变量有着特殊的含义,用户不可以更改,所有的预定义变量都由 `<article class=\”_2rhmJa\” 符号和另外一个符号组成,常用的预定义特殊变量如下:
$! 上一个后台命令对应的进程号$? 上一个命令的退出状态,为十进制数字,如果返回为0,则代表执行成功,则否为不成功。$ 当前的进程号PID
以上变量请配合
echo
使用,例如:
echo $!echo $echo $?
位置变量
# 位置变量也属于预定义的变量# 位置变量真对于脚本来说# $0 程序名称,脚本的文件名# $1 执行脚本时,跟到脚本后面的第一个参数# $2...# $# 位置变量的个数# $@ 所有的位置变量的值echo \"$0\"echo \"$1\"echo \"$2\"echo \"位置参数的个数$#\"echo \"所有的参数$@
假如脚本中使用了一个位置变量,但是执行脚本的时候,传入了多个参数,程序并不会报错。
假如脚本中使用了多个位置变量,比如脚本内容如下 :
echo \"$0\"echo \"$1\"echo \"$2\"echo \"位置参数的个数$#\"echo \"所有的参数$@\"\"
但是执行的时候只传入了一个参数,同样不会报错,并且
$2
不会被赋值
有什么用呢?比如我想计算 任意 2 个数的商数和余数,可以这么写
some2number.sh
脚本内容:
echo \"商数: $(( $1 / $2 ))\"echo \"余数: $(( $1 % $2 ))\"
使用:
[root@kube-master script]# sh some2number.sh 10 3商数: 3余数: 1
易读版本
[root@kube-master script]# cat some2number2.shn1=$1n2=$2echo \"商数: $(( n1 / n2 ))\"echo \"余数: $(( n1 % n2 ))\"
使用
[root@kube-master script]# sh some2number2.sh 20 8商数: 2余数: 4
五、变量键盘读取、数组与宣告: read, array, declare
read
[root@www ~]# read [-pt] variable选项不参数:-p :后面可以接提示字符!-t :后面可以接等待的『秒数!』这个比较有趣~不会一直等待使用者啦!
例如:
#vi read.shread -p “请输入你的姓名” nameecho \"你的姓名是: $name\"
数组 (array) 变量类型
数组的索引只能是数字
var=(wukong bajie shaseng)echo ${var [2]} //显示某一个输出 bajieecho ${var [*]} //显示数组的全部内容输出 wukong bajie shaseng
declare 声明关联数组
数组的索引可以是普通字符串
declare -A
declare -A info 声明关联数组info[name]=shark 添加索引 name 其对应的值是 sharkinfo=([age]=18 [hobby]=\"mv rmb\") 一次添加多个索引和值echo ${info[name]} 显示出名字echo ${info[@]} 显示出所有的值echo ${!info[@]} 显示所有的索引
六、变量内容的删除、取代与替换
• 删除
先让小写的 path 自订变量设置的与 PATH 内容相同[dmtsai@study ~]$ path=${PATH}[dmtsai@study ~]$ echo ${path}/usr/local/bin:/usr/bin:/usr/local/sbin:/usr/sbin:/home/shark/bin
- 删除最后一个
echo ${path%:*}
%
从字符串的后面开是进行匹配删除,但匹配到第一个时就结束匹配。
:*
就是需要进行匹配的字符串 ,
:
是普通字符串
*
是通配符,代表任何数量的任意字符串。
所以下面的红色字体的字符串将会匹配后进行删除
/usr/local/bin:/usr/bin:/usr/local/sbin:/usr/sbin
:/home/shark/bin
- 删除第一个
接着上例的变量进行操作
echo ${path#*:}
#
是从前面开始匹配删除,匹配到道理和上面的
%
同样的道理
${path#*:}
就是 从变量值的最前面开始匹配,直到遇到第一个
:
结束
所以下面的红色字体会被匹配后删除
/usr/local/bin:
/usr/bin:/usr/local/sbin:/usr/sbin:/home/shark/bin
- 取代和替换
以上不需要都掌握,需要记住下面这个
var=${str:-expr}
str
和
var
可以是相同的字符串。比如
inpu_date=${inpu_date:-expr}
expr
可以是任意合法的表达式,比如一个字符串,一个子 shell 的命令
$(date +%F)
示例,假如希望用户输入一个日期,当用户没有输入时,就给一个默认值,这里给的是当天的日期
read -p \"输入日期>:\" input_dateinput_date=${input_date:-$(date +%F)}echo \"当前的日期是:${input_date}\"
七、时间运算
// 计算 3 小时之后是几点几分date +%T -d \'3 hours\'// 任意日期的前 N 天,后 N 天的具体日期date +%F -d \"20190910 1 day\"date +%F -d \"20190910 -1 day\"// 计算两个日期相差天数, 比如计算生日距离现在还有多少天d1=$(date +%s -d 20180728)d2=$(date +%s -d 20180726)echo $(( (d1-d2) / 86400 ))# 输出2,这里给的是当天的日期
read -p “输入日期>:” input_date
input_date=KaTeX parse error: Expected \’}\’, got \’EOF\’ at end of input: {input_date:-(date +%F)}
echo “当前的日期是:${input_date}”
# 七、时间运算
// 计算 3 小时之后是几点几分
date +%T -d ‘3 hours’
// 任意日期的前 N 天,后 N 天的具体日期
date +%F -d “20190910 1 day”
date +%F -d “20190910 -1 day”
// 计算两个日期相差天数, 比如计算生日距离现在还有多少天
d1=(date+d2=(date +%s -d 20180728)d2=(date+d2=(date +%s -d 20180726)
echo $(( (d1-d2) / 86400 ))
输出
2