1、 数组
变量:存储单个元素的内存空间
数组:存储多个元素的连续的内存空间,相当于多个变量的集合
数组名和索引
索引:编号从0 开始,属于数值索引
注意:索引可支持使用自定义的格式,而不仅是数值格式,即为关联索引,bash4.0版本之后开始支持。而且bash 的数组支持稀疏格式(索引不连续)
1)、定义数组
声明数组:
declare -a ARRAY_NAME
declare -A ARRAY_NAME:关联数组
数组元素的赋值:
(1) 一次只赋值一个元素;
若元素里面没有空格,双引号可以不写
ARRAY_NAME[INDEX]=VALUE
arr[0]=123
arr[1]=abc
week[0]=Sunday
day[4]="third day"
(2) 一次赋值全部元素:
各个元素间以空格隔开,且若元素里面没有空格,可以不使用双引号
ARRAY_NAME=("VAL1" "VAL2" "VAL3" …)
arr=(1 2 3 4 5 abc df)
(3)只赋值特定元素也即是上面讲的稀疏格式:
ARRAY_NAME=([0]="VAL1" [3]="VAL2" …)
Arr=([0]=1 [2]=a [5]=mn)
(4)交互式数组值对赋值
read -a ARRAY
[root@centos7 ~]# read -a arr
a b c
[root@centos7 ~]# echo ${arr}
a
[root@centos7 ~]# echo ${arr[0]}
a
[root@centos7 ~]# echo ${arr[1]}
b
[root@centos7 ~]# echo ${arr[2]}
c
[root@centos7 ~]# echo ${arr[*]}
a b c
[root@centos7 ~]# echo ${arr[@]}
a b c 引用数组中的所有元素
[root@centos7 ~]# echo ${arr[@]:1:1}
b
[root@centos7 ~]# echo ${arr[@]:1}
b c 数组切片
[root@centos7 ~]# arr[${#arr[*]}]=d向数组中追加元素
[root@centos7 ~]# echo ${arr[@]}
a b c d
2)、引用数组
引用数组元素:${ARRAY_NAME[INDEX]}
注意:省略[INDEX] 表示引用下标为0的元素
数组的长度( 数组中元素的个数):
${#ARRAY_NAME[*]}
${#ARRAY_NAME[@]}
示例1:生成10 个随机数保存于数组中,并找出其最大值和最小值
#!/bin/bash
declare -a rand
declare -i max=0 在算术运算里面有说明,声明数值变量
for i in {0..9}; do
rand[$i]=$RANDOM
echo ${rand[$i]}
[ ${rand[$i]} -gt $max ] && max=${rand[$i]}
done
echo "the max is: $max"
示例2:写一个脚本,定义一个数组,数组中的元素是/var/log 目录下所有以.log 结尾的文件;要统计其下标为偶数的文件中的行数之和
[root@centos7 ~]# wc -l /etc/passwd
52 /etc/passwd
#!/bin/bash
declare -a files
files=(/var/log/*.log)数组匹配的是文件名,支持通配符
declare -i lines=0
for i in $(seq 0 $[${#files[*]}-1]); do
if [ $[$i%2] -eq 0 ];then
let lines=$[lines+$(wc -l ${files[$i]} | cut -d' ' -f1)]
fi
done
echo "the lines is: $lines"
3)、数组数据处理
引用数组中的元素:
所有元素:${ARRAY[@]}, ${ARRAY[*]}
数组切片:${ARRAY[@]:offset:number}
offset: 要跳过的元素个数
number: 要取出的元素个数
取偏移量之后的所有元素:${ARRAY[@]:offset}
向数组中追加元素:
ARRAY[${#ARRAY[*]}]
[root@centos7 ~]# arr[${#arr[*]}]=d 标记部分为数组的长度
删除数组中的某元素:导致稀疏格式
unset ARRAY[INDEX]
关联数组-必须先声明再调用:
#declare -A ARRAY_NAME
#ARRAY_NAME=([idx_name1]=('val1' [idx_name2]='val2‘…)
#arr=([a]=aaa [b]=bb [c]=5)
#echo ${arr[a]}
2、 字符串处理-bash 的字符串处理工具
1)、字符串切片
${#var}: 返回字符串变量var的长度
${var:offset}: 返回字符串变量var中从第offset个字符后(不包括第offset 个字符)的字符开始,到最后的部分,offset 的取值在0到${#var}-1之间
${var:offset:number}:返回字符串变量var中从第offset个字符后(不包括第offset 个字符)的字符开始 ,长度为number的部分
${var: -lengh}:取字符串的最右侧几个字符,比如#echo ${var: -3}
# echo ${var:1: -3}意思是掐头一个去尾两个,取中间部分
注意:冒号后必须有一空白字符
2)、基于模式取子串:
${var#*word}:其中word可以是指定的任意字符
功能:自左而右,查找var变量所存储的字符串中,第一次出现的word, 删除自字符串开头至第一次出现word字符之间(包括第一次出现的word字符)的所有字符
${var##*word}:同上,不同的是,删除的是自字符串开头至最后一次出现word 字符之间(包括最后一次出现的word字符)的所有内容
file="/var/log/messages"
${file##*/}: messages
${var%word*}:其中word 可以是指定的任意字符
功能:自右而左,查找var 变量所存储的字符串中,第一次出现的word, 删除自字符串最右侧的字符向左至第一次出现word 字符之间(包括第一次出现的word字符)的所有字符;
file="/var/log/messages"
${file%/*}: /var/log
${var%%word*}:同上,只不过是删除自字符串最右侧的字符向左至最后一次出现word 字符之间(包括第一次出现的word字符)的所有字符
示例:url=http://www.magedu.com:80
${url##*:} 80
${url%%:*} http
3)、查找替换
${var/pattern/substi}:查找var所表示的字符串中,第一次被pattern 所匹配到的字符串以substi替换之
${var//pattern/substi}: 查找var所表示的字符串中,所有能被pattern 所匹配到的字符串以substi替换之
${var/#pattern/substi}:查找var所表示的字符串中,行首被pattern 所匹配到的字符串以substi 替换之
${var/%pattern/substi}:查找var所表示的字符串中,行尾被pattern 所匹配到的字符串以substi替换之
4)、查找并删除
${var/pattern}:查找var 所表示的字符串中,删除第一次被pattern 所匹配到的字符串
${var//pattern}:所有
${var/#pattern}:首行
${var/%pattern}:行尾
[root@centos7 ~]# grep root /etc/passwd|head -1
root:x:0:0:root,,62895000:/root:/bin/bash
[root@centos7 ~]# var=`grep root /etc/passwd|head -1`
[root@centos7 ~]# echo ${var/root}
:x:0:0:root,,62895000:/root:/bin/bash
5)、字符大小写转换
${var^^}:把var中的所有小写字母转换为大写
${var,,}:把var中的所有大写字母转换为小写
3、 变量赋值
${var:-value}:如果var为空或未设置,那么返回value ;否则,则返回var的值
例如var变量未设置
[root@centos7 ~]# releasename=${var:-rhel}
[root@centos7 ~]# echo $releasename
rhel
${var:+value}:如果var不空,则返回value
${var:=value}:如果var 为空或未设置,那么返回value ,并将value 赋值给var ;否则,则返回var的值
${var:?error_info}:如果var为空或未设置,那么在当前终端打印error_info ;否则,则返回var的值
为了在脚本程序中使用事先配置好的变量,可仿照存放函数的文本文件定义存放变量的文本文件实现变量赋值:
(1)定义文本文件,每行定义变量
(2)在脚本中source此文件即可使用定义好的变量,同函数调用
4、高级变量用法-有类型变量
Shell变量一般是无类型的,但是bash Shell提供了declare和typeset 两个命令用于指定变量的类型,两个命令是完全等价的
declare [ 选项] 变量名
-r 将变量设置为只读属性
-i 将变量定义为整型数
-a 将变量定义为数组
-A 将变量定义为关联数组
-f 显示此脚本以前定义过的所有函数名及其内容
#declare –f
-F 仅显示此脚本以前定义过的所有函数名
#declare -F
-x 将变量声明为环境变量
-l 将变量值转为小写字母
-u 将变量值转为大写字母
[root@centos7 ~]# declare -l var=CENTOSaabb
[root@centos7 ~]# echo $var
Centosaabb
5、间接变量引用
如果第一个变量的值是第二个变量的名字,从第一个变量引用第二个变量的值就称为间接变量引用
variable1=variable2
variable2=value
variable1的值是variable2 ,而variable2 又是变量名,variable2 的值为value ,间接变量引用是指通过variable1获得变量值value 的行为
bash Shell提供了两种格式实现间接变量引用
#eval tempvar=\$$variable1
#tempvar=${!variable1}
示例:
[root@server ~]# N=NAME
[root@server ~]# NAME=wangxiaochun
或者
#echo ${!N}
wangxiaochun
或者
[root@server ~]# N1=${!N}
[root@server ~]# echo $N1
wangxiaochun
或者
#eval echo \$$N
wangxiaochun
或者
[root@server ~]# eval N2=\$$N
[root@server ~]# echo $N2
wangxiaochun
6、eval 命令
eval命令将会首先扫描命令行进行所有的置换,然后再执行该命令;该命令适用于那些一次扫描无法实现其功能的变量;该命令对变量进行两次扫描 。
示例-运行变量V1中存储的命令;如果变量值不是命令,会出现执行错误
[root@server ~]# V1=pwd
[root@server ~]# echo $V1
pwd
[root@server ~]# eval $V1
/root
等同
#`echo $V1`
/root
7、创建临时文件
mktemp 命令:创建的临时随机文件避免冲突
mktemp [OPTION]… [TEMPLATE] 其中TEMPLATE的格式是:filename.XXX,X至少要出现三个
OPTION:
-d:创建临时目录
–tmpdir=/DIR:指明临时文件所存放的目录位置
实例:
#mktemp –tmpdir=/testdir test.XXXXXX
#mktemp /testdir/file.XXX
#mktemp -d /tmp/test.XXX
8、安装复制文件
install命令-复制文件时会默认加上755权限,一般用于复制脚本等程序文件:
install [OPTION]… [-T] SOURCE DEST 复制单文件至目标文件或者文件夹
install [OPTION]… SOURCE… DIRECTORY 复制多个文件至目标文件夹
install [OPTION]… -t DIRECTORY SOURCE…
install [OPTION]… -d DIRECTORY… 创建空目录
选项:
-m MODE,默认755,指定权限
-o OWNER,指定所属主
-g GROUP,指定所属组
[root@centos7 ~]# install -m 744 -o wang f2 /testdir/f3
[root@centos7 ~]# ll f2 /testdir/f3
–w——-. 1 root root 10 Aug 15 10:35 f2
-rwxr–r–. 1 wang root 10 Aug 24 21:49 /testdir/f3
[root@centos7 ~]# install f2 /testdir
[root@centos7 ~]# ll f2 /testdir/f2
–w——-. 1 root root 10 Aug 15 10:35 f2
-rwxr-xr-x. 1 root root 10 Aug 24 21:40 /testdir/f2
9、防止扩展
反斜线\会使随后的字符按原意解释
[root@centos7 ~]# var=aaa
[root@centos7 ~]# echo this output is:$var
this output is:aaa
[root@centos7 ~]# echo this output is:\$var
this output is:$var
[root@centos7 ~]# var=aaa
[root@centos7 ~]# [ $var == aaa ] && echo $? && exit
0
logout
[root@centos7 ~]# var=aaa
[root@centos7 ~]# [ $var == aaa ] && {echo xxx;exit;}
-bash: syntax error near unexpected token `}',echo前面应有一个空格
[root@centos7 ~]# [ $var == aaa ] && { echo xxx;exit;}
xxx
logout 大括号是匿名块或者匿名函数的作用
[root@centos7 ~]# var=aaa
[root@centos7 ~]# [ $var == bbb ] && echo $? && exit 无任何输出也不退出
[root@centos7 ~]# var=aaa
[root@centos7 ~]# [ $var == bbb ] && echo $?;exit
logout 此处后面的exit是独立的,与其前面的判断无关,所以exit总会执行进而退出
加引号来防止扩展
单引号(’ )防止所有扩展,即变量值不替换
双引号(” )也防止所有扩展,但是以下情况例外:
$ (美元符号) - 变量扩展
` (反引号) - 命令替换
\ (反斜线) - 禁止单个字符扩展
! (叹号) - 历史命令替换
10、bash 的配置文件
1)、按生效范围划分,存在两类:
全局配置:
/etc/profile
/etc/profile.d/*.sh
/etc/bashrc
个人配置:
~/.bash_profile
~/.bashrc
2)、shell 两种登录方式
交互式登录:
(1)直接通过终端输入账号密码登录
(2)使用“su – UserName”切换的用户
触发执行配置文件的类型与顺序:
/etc/profile –> /etc/profile.d/*.sh –>~/.bash_profile –> ~/.bashrc –> /etc/bashrc
若有设置相同,后执行的会覆盖先执行的程序
非交互式登录:
(1)su UserName
(2)图形界面下打开的终端
(3)执行脚本
触发执行配置文件类型与顺序:
~/.bashrc –> /etc/bashrc –> /etc/profile.d/*.sh
11、配置文件的profile 类和bashrc类
Profile类:为交互式登录的shell 提供配置
全局:/etc/profile, /etc/profile.d/*.sh
个人:~/.bash_profile
功用:用于定义环境变量、运行命令或脚本
bashrc 类:为非交互式和交互式登录的shell 提供配置
全局:/etc/bashrc
个人:~/.bashrc
功用:定义命令别名和函数、定义本地变量
编辑配置文件生效
修改profile 和bashrc 文件后需生效
两种方法:重新启动shell进程或者 .或source
例如:
# . ~/.bashrc,点号后有一个空格
# source ~/.bashrc
原创文章,作者:18612763863,如若转载,请注明出处:http://www.178linux.com/42380