一、什么是变量
简单说就是让一个特定的字符串代表不固定的内容
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/bin
PATH=${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=shark
export name
bash
echo $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.sh
n1=$1
n2=$2
echo "商数: $(( 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.sh
read -p “请输入你的姓名” name
echo "你的姓名是: $name"
数组 (array) 变量类型
数组的索引只能是数字
var=(wukong bajie shaseng)
echo ${var [2]} //显示某一个
输出 bajie
echo ${var [*]} //显示数组的全部内容
输出 wukong bajie shaseng
declare 声明关联数组
数组的索引可以是普通字符串
declare -A
declare -A info 声明关联数组
info[name]=shark 添加索引 name 其对应的值是 shark
info=([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_date
input_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
网友评论