基本数据类型与值操作
一 数据类型介绍
什么是数据?为何要有多种类型的数据?
#数据即变量的值,如age=18,18则是我们保存的数据。
#变量的是用来反映/保持状态以及状态变化的,毫无疑问针对不同的状态就应该用不同类型的数据去标识
shell是一门解释型、弱类型、动态语言
概括地说,编程语言的划分方式有以下三种
1、编译型or解释型
2、强类型or弱类型
2.1 强类型语言: 数据类型不可以被忽略的语言
即变量的数据类型一旦被定义,那就不会再改变,除非进行强转。
在python中,例如:name = 'egon',这个变量name在被赋值的那一刻,数据类型就被确定死了,是字符型,值为'egon'。
2.2 弱类型语言:数据类型可以被忽略的语言
比如linux中的shell中定义一个变量,是随着调用方式的不同,数据类型可随意切换的那种,即shell对数据类型的概念没有那么强的要求
3、动态型or静态型
3.1 动态语言 :运行时才进行数据类型检查
即在变量赋值时,就确定了变量的数据类型,不用事先给变量指定数据类型
3.2 静态语言:需要事先给变量进行数据类型定义
所以综上所述,shell是一门解释型的弱类型动态语言
二 基本数据类型
数字
#int整型
定义:age=10
用于标识:年龄,等级,身份证号,qq号,个数
#float浮点型
定义:salary=3.1
用于标识:工资,身高,体重
字符串
#在shell中,加了引号的字符就是字符串类型
定义:name='dan'
用于标识:描述性的内容,如姓名,性别,国籍,种族
# 注意1:字符串包含空格必须加引号
[root@dan ~]# msg="hello dan"
[root@dan ~]# msg=hello dan
bash: dan: 未找到命令...
# 注意2:连续的字符不加引号包含也是可以的,但我们还是强烈建议加上引号,规范一些
[root@dan ~]# msg=hello
[root@egon ~]# echo $msg
hello
# 注意3:单引号与双引号是不同的
" " 弱引用,引号的特殊字符有意义
' ' 强引用,引号内所有特殊字符都取消意义
[root@dan ~]# name=“dan”
[root@dan ~]# echo "${name} is good"
dan is good
[root@dan ~]# echo '${name} is good'
${name} is good
shell是弱类型语言
[root@dan ~]# x=10
[root@dan ~]# y="3"
[root@dan ~]# expr $x + $y
13
数组介绍
# 1、什么是数组?
数组就是一系列元素的集合,一个数组内可以存放多个元素
# 2、为何要用数组?
我们可以用数组将多个元素汇总到一起,避免单独定义的麻烦
数组分为两种
- 普通数组:只能使用整数作为数组索引
- 关联数组:可以使用字符串作为数组索引,需要用declare -A声明
普通数组
=================声明普通数组=================
# 方式一:array=(元素1 元素2 元素3)
array=(dan 18 male)
# 方式二:array=([key1]=value1 [key2]=value2 [key3]=value3)
array=([0]=111 [1]="two" [2]=333)
# 方式三:依次赋值
array_new[0]=111
array_new[1]=222
array_new[2]="third"
# 方式四:利用执行命令的结果设置数组元素:array=($(命令)) 或者 array=(`命令`)
该方式会将命令的结果以空格为分隔符切成多个元素然后赋值给数组
[root@aliyun ~]# ls /test
a.txt b.txt
[root@aliyun ~]# array3=(`ls /test`)
[root@aliyun ~]# declare -a |grep array3
declare -a array3='([0]="a.txt" [1]="b.txt")'
# ps:查看声明过的数组
declare -a
=================访问普通数组=================
[root@dan ~]# ip_array=(1.1.1.1 2.2.2.2 3.3.3.3)
# 正向索引
[root@dan ~]# echo ${ip_array[0]}
1.1.1.1
[root@dan ~]# echo ${ip_array[1]}
2.2.2.2
[root@dan ~]# echo ${ip_array[2]}
3.3.3.3
# 负向索引
[root@dan ~]# echo ${ip_array[-1]}
3.3.3.3
[root@dan ~]# echo ${ip_array[-2]}
2.2.2.2
[root@dan ~]# echo ${ip_array[-3]}
1.1.1.1
关联数组
=================声明关联数组=================
[root@aliyun ~]# declare -A info
[root@aliyun ~]# info["name"]="dan"
[root@aliyun ~]# info["age"]=18
[root@aliyun ~]# info["gender"]="male"
[root@aliyun ~]# declare -A |grep info
declare -A info='([gender]="male" [name]="dan" [age]="18" )'
[root@aliyun ~]# echo ${info[*]}
male dan 18
[root@aliyun ~]# echo ${info["name"]}
=================访问关联数组=================
[root@dan ~]# info=([0]="dan" ["age"]=18 ["gender"]="male")
[root@dan ~]# echo ${info[0]}
egon
[root@dan ~]# echo ${info["age"]}
18
[root@dan ~]# echo ${info["gender"]}
male
ps:bash shell只支持一维数组,但数组元素个数没有限制。
三 变量值操作
3.1 获取变量值的长度
[root@localhost ~]# echo ${#url}
15
# 企业面试题:已知变量msg='hello world!',请统计出变量中包含的字符数量
# 方法一:
[root@dan /]# echo ${#msg}
12
# 方法二:
[root@dan /]# echo $msg | wc -L
12
# 方法三:
[root@dan /]# echo $msg|awk '{print length}'
12
# 方法四:
[root@dan ~]# expr length "$msg" #length是一个函数,注意因为msg的值有空格,所以$msg必须用引号包含
12
3.2 切片
${paramter:offset:length}
[root@dan /]# msg="abcdef"
[root@dan /]# echo ${msg:3} # 从3号索引开始,一直到最后
def
[root@dan /]# echo ${msg:3:2} # 从3号索引开始,往后数2个字符
de
[root@dan /]# echo ${msg::3} # 从0开始,往后数3个字符
abc
3.3 截断
# =================》一、砍掉左边的字符《=================
# 1.1 简单使用
[root@dan ~]# url="www.sina.com.cn"
[root@dan ~]# echo ${url#www.}
sina.com.cn
# 1.2 结合*=》非贪婪,默认情况下*是非贪婪,尽可能地少“吃”字符
[root@dan ~]# echo ${url#*w}
ww.sina.com.cn
# 1.3 结合*=》贪婪,尽可能地多“吃”字符
[root@dan ~]# echo ${url##*w} # *会尽可能多地吃掉字符,一直匹配到最远的那个w才停下来
.sina.com.cn
# =================》二、砍掉右边的字符《=================
# 1.1 简单使用
[root@dan ~]# url="www.sina.com.cn"
[root@dan ~]# echo ${url%.cn}
www.sina.com
# 1.2 结合*=》非贪婪
[root@dan ~]# echo ${url%.*}
www.sina.com
# 1.3 结合*=》贪婪
[root@dan ~]# echo ${url%%.*}
www
# =================》三、应用示例《=================
[root@dan ~]# hostname
dan.xxx.com
[root@dan ~]# echo $HOSTNAME
dan.xxx.com
[root@dan ~]# echo ${HOSTNAME%.*}
dan.xxx
[root@dan ~]# echo ${HOSTNAME%%.*}
dan
3.4 内容的替换
[root@dan ~]# url="www.sina.com.cn"
[root@dan ~]# echo ${url/sina/baidu}
www.baidu.com.cn
[root@dan ~]# echo ${url/n/N}
www.siNa.com.cn
[root@egon ~]# echo ${url//n/N} # 贪婪
www.siNa.com.cN
# 应用示例:批量修改文件名称
[root@dan shell]# touch egon_2020_{01..05}_linux.txt
[root@dan shell]# ls
[root@dan shell]# for i in `ls *linux.txt`;do mv $i ${i/_linux/};done
3.5 变量的替代
${x:-临时变量信息}
${x:=新的变量信息}
${x:?没有设置变量提示信息}
${x:+有设置变量提示信息}
#1、${parameter-word}: 当调取变量没有定义过, 就返回word字符串信息
[root@dan ~]# unset name
[root@dan ~]# echo ${name}
[root@dan ~]# echo ${name-"dan"} # 没有定义过变量name,则使用-后的值
dan
[root@dan ~]# gender= # 定义过变量了,则使用变量的原值,哪怕变量的值为空值
[root@dan ~]# echo ${gender-"male"}
[root@dan ~]# age=18
[root@dan ~]# echo ${age-19} # 定义过变量了,则使用变量的原值
18
#2、${parameter:-word}: 当调取变量信息值为空时或未定义变量, 就返回word字符串信息
[root@dan ~]# unset x
[root@dan ~]# unset y
[root@dan ~]# unset z
[root@dan ~]# echo ${x:-aaa} # 没有定义过变量x,则使用-后的值
aaa
[root@dan ~]# y=
[root@dan ~]# echo ${y:-aaa} # 定义过变量y,但变量y的值为空值,则使用-后的值
aaa
[root@dan ~]# z=333
[root@dan ~]# echo ${z:-aaa} # 定义过变量了,并且变量有一个非空的原值,则使用变量的原值
333
#3、{parameter:=word}:当调取变量信息值为空时或未定义,则设置指定字符串为新的变量值
[root@dan /]# unset x
[root@dan /]# echo ${x:=123}
123
[root@dan /]# echo $x
123
#4、${parameter:?word}:当调取变量信息值为空时或未定义,指定为赋值的错误提示信息
[root@egon /]# unset x
[root@egon /]# echo ${x:?该变量没有定义过}
-bash: x: 该变量没有定义过
#5、${parameter:+word}:当调取变量信息值为空时或未定义,不做任何处理,否则word字符串将替代变量值
[root@egon /]# unset x
[root@egon /]# echo ${x:+哈哈哈}
[root@egon /]# x=123
[root@egon /]# echo ${x:+哈哈哈}
哈哈哈
3.6 let
# (1) 变量的值
[root@egon ~]# j=1
[root@egon ~]# let ++j
[root@egon ~]# echo $j
2
[root@egon ~]#
# (2) 表达式的值
[root@egon ~]# unset i
[root@egon ~]# unset j
[root@egon ~]#
[root@egon ~]# i=1
[root@egon ~]# j=1
[root@egon ~]#
[root@egon ~]# let x=i++ # 先把i赋值给x,然后再++
[root@egon ~]# let y=++j # 先++j,然后再把j的结果赋值给y
[root@egon ~]# echo $i
2
[root@egon ~]# echo $j
3
[root@egon ~]# echo $x
1
[root@egon ~]# echo $y
2
3.7 取命令的结果赋值给变量:
# ``与$()
` ` 命令替换 等价于 $() 反引号中的shell命令会被先执行
[root@localhost ~]# touch `date +%F`_file1.txt
[root@localhost ~]# touch $(date +%F)_file2.txt
[root@localhost ~]# disk_free3="df -Ph |grep '/$' |awk '{print $4}'" # 错误
[root@localhost ~]# disk_free4=$(df -Ph |grep '/$' |awk '{print $4}') # 正确
[root@localhost ~]# disk_free5=`df -Ph |grep '/$' |awk '{print $4}'` # 正确