关于shell脚本编程基础第五篇
shellb编程基础第五篇本章内容:数组
变量:存储单个元素的内存空间
数组:存储多个元素的连续的内存空间,相当于多个变量的集合。
索引:编号从0开始,属于数值索引
注意:索引可支持使用自定义的格式,而不仅是数值格式,即为关联索引,bash4.0版本之后开始支持。
bash的数组支持稀疏格式(索引不连续)
数组名[索引]
${数组名[索引]}
定义数组:
声明数组:declare -a NAME 声明索引数组
declare -A NAME 声明关联数组
数组中元素赋值方式
一次只赋值一个元素:
一次赋值全部元素
只赋值特定元素
交互式数组值赋值 read -a 数组陈列
===================================================================
一次性元素赋值方式:
# declare -a name 声明索引数组名称
# name=xiaobo 数组元素
# name=laozai 数组元素
# echo ${name
[*]} 调用全部元素
xiaobo laozai
# echo ${name} 调用0的元素
xiaobo
# echo ${name} 调用1的元素
laozai
===================================================================
一次赋值给全部元素:
# name=("xiaobo" "laozai") 数组名称=元素列表
# echo ${name} 输出打印数组0的参数
xiaobo
# echo ${name} 输出打印数组1的参数
laozai
# echo ${name
[*]} 输出打印全部元素
xiaobo laozai
====================================================================
只给特定元素:
# name=(="xiaobo" ="laozai") 数组名称=特定给元素
# echo ${name}
xiaobo
# echo ${name}
laozai
# echo ${name
[*]}
xiaobo laozai
====================================================================
命令行举例交互式数组赋值:
# read -a name 读取数组 名称
xiaomag xiaobo laozai 写入对应元素
# echo ${name} 0就是第一个元素
xiaomag
# echo ${name} 1就是第二个元素
xiaobo
# echo ${name} 2就是第三个元素
laozai 这叫做索引数组
======================================================================
# name=xiaobo
# echo ${name}
xiaobo
这种叫做关联数组
=====================================================================
引用数组中的元素$ {name}
注意 :引用时只给数组名,表示引用下标为0的元素
数组的长度,(数组中元素的个数)
${#name
[*]}
${#name[@]}就加上括号表示元素中的个数 ,不加表示第一个元素字符中的个数
=============================================================
练习
# vim shuzusuiji.sh 编辑脚本名称
#!/bin/bash
#
#uesr:Compro
#数组生成10个随机数保存与数组中,并找出其最大值和最小值
declare -a name 声明索引数组 name
declare -i max=0 声明整数属性
for i in {0..9};do 循环 变量i的值,0-9
name[$i]=$RANDOM 数组赋值上条命令的值=随机
echo ${name[$i]} 打印数组的随机值
[ ${name[$i]} -gt $max ] && max=${name[$i]} 条件判断打印的值是否大于0.如果大于0的话就执行 max=数组$i的元素
done
echo "Max:$max" 输出打印上面的$max的值
===================================================
练习
# vim oushuzhihe.sh
#!/bin/bash
#
#user:Compro
#写一个脚本,定义一个数组,数组中的元素是/var/log目录下所有以.log结尾的文件;
要统计其下标为偶数的文件中的行数之和
declare -a files 声明索引数组 files
files=( /var/log/*.log ) 数组元素= 路径
declare -i lines=0 声明整数属性
for i in $(seq 0 $[${#files
[*]}-1]);do 循环 变量i的值 ,追加数组元素
if [ $[$i%2] -eq 0 ];then 分支判断变量值等于0
let lines+=$(wc -l ${files[$i]} | cut -d' ' -f1 )计算变量值 抽取以空白符为分隔符的第1列的值
fi
done
echo "Lines: $lines." 输出打印结果
==============================================================
引用数组中的元素
所有元素:${数组[@]},${数组
[*]}
数组切片:${数组[@]}:offset:number
offset 要跳过的元素个数
number 要取出的元素个数
取偏移量之后的所有元素${数组[@]:offset}
向数组中最加元素:
数组名称[${#数组元素
[*]}]
删除数组中的某元素:导致稀疏格式
unset 数组[索引]
关联数组:
declare -A 数组名称 必选先声明,在调用
数组名称=([元素1]='val1' [元素2] 'val2'...)
========================================================================
bash 的字符串处理工具
字符串切片:
${#var} 返回字符串变量var的长度
${var:offset} offset返回字符串的变量var中第offset个字符后(不包括第offset个字符)的字符开始,
到最后的部分offset的取值在0到 ${#var}-1 之间(bash4.2后,允许为负值)
${var:offset:number}返回字符串变量var中从第offset个字符 后(不包括第offset个字符)
的字符开始,长度为number的部分
${var: -数字} 取字符串的最右侧几个字符
注意:冒号后面必须有一个空白字符
# var=`echo {a..z}`
# v=`echo $var |tr -d " "`
# echo $v
abcdefghijklmnopqrstuvwxyz
# echo ${#v}
26
# echo ${v:3}
defghijklmnopqrstuvwxyz
# echo ${v:3:5}
defgh
# echo ${v: -3}
xyz
=====================================================================
基于模式取字符串:
${var#*word} 其中word可以是指定的任意字符
功能:自左而右,查找var变量所存储的字符串中,第一次出现word,
删除字符串开头至第一次出现word字符之间的所有字符
~]# echo ${v#*m}
nopqrstuvwxyz
${var##*word} 同上,不同的是,删除的是字符串开头至
最后一次由word指定的字符之间的所有内容
~]# file="/var/log/messages"
~]# echo ${file##*/}
messages
${var%word*} 其中word可以是指定的任意字符
功能:自右而左,查找var变脸所存储的字符串中,第一次出现的word,
删除字符串最后一个字符向左至第一次出现word字符之间的所有字符
~]# file="/var/log/messages"
~]# echo ${file%/*}
/var/log
${var%%word*} 同上,只不过删除字符串最右侧的字符向左至
最后一次出现word字符之间的所有字符
url=aaaa:5556:222
~]# echo ${url##*:}
222 最右侧的
~]# echo ${url%%:*}
aaaa 最左侧的
# var=hacker:x:1005:1005:hacker,,38763415:/home/hacker:/bin/bash
# echo $var
hacker:x:1005:1005:hacker,,38763415:/home/hacker:/bin/bash
# echo ${var#*x}
:1005:1005:hacker,,38763415:/home/hacker:/bin/bash
==============================================================================
查找替换:
${var/pattern/substi} : 查找var所表示的字符串,第一次被pattern所匹配到的字符串,以路径替换之
${var//pattern/substi} : 查找var所表示的字符串,所有被pattern所匹配到的字符串,以路径替换之
${var/#pattern/substi} : 查找var所表示的字符串,行首被pattern所匹配到的字符串,以路径替换之
${var/%pattern/substi} : 查找var所表示的字符串,行尾被pattern所匹配到的字符串,以路径替换之查找并删除:
${var/pattern} 查找var所表示的字符串中,删除第一次被pattern所匹配到的字符串
${var//pattern}所有
${var/#pattern}行首
${var/%pattern}行尾
字符大小写转换:
${var^^}把var中的所有小写字母转换为大写
${var,,}把var中的所有大写字母转换为小写
变量赋值:
${var:-value} 如果var为空或未设置,那么返回value;否则,则返回var的值
${var:+value} 如果var不空,则返回value
${var:=value} 如果var为空或未设置,那么返回value,并将value 赋值给var;f否则,则返回var的值
${var:?error_info} 如果var为空或未设置,那么返回error_info;否则,
则返回var的值为脚本程序使用配置文件,实现变量赋值
定义文本文件,每行定义“name=value”
在脚本中source此文件即可
高级变量用法-有类型变量:
shell变量一般是无类型的,但是bash shell提供了declare和
typeset两个命令用于指定变量的类型,两个命令是完全等价的
declare [选项] 变量名
-r name 将变量设置为只读属性
-i name 将变量定义为整型数
-a name 将变量定义为数组
-f name 显示此脚本前定义过的所有函数名及其内容
-Fname 仅显示此脚本前定义过的所有函数名
-x name 将变量声明为环境变量
-l name 将变量转为小写字母
==============================================================
间接变量引用:
如果第一个变量的值是第二个变量的名字,
从第一个变量引用第二个变量的值就称为间接变量引用
var1=var2
var2=var3
var1的是是var2,而var2又是变量名,var2的值为var3,间接变量引用是指通过var1获得不变量值var3的行为
# echo ${!var1} 第一种间接引用格式
var3
bash shell提供了两种格式实现间接变量引用
# n=name
# name=xiaomag
# n1=${!n}
# echo $n1
xiaomag
# eval n3=\$$v1 第二种间接引用格式
# echo $n3
xiaomag
===============================================================
eval命令
将会首先扫描命令行进行所有的置换,然后在执行该命令。
改名了适用于那些一次扫描无法实现其功能的变量,该命令对变量进行两次扫描
示例:
# cmd=whoami
# echo $cmd
whoami
# eval $cmd
root
=================================================================
创建临时文件
mktemp命令创建的临时文件可避免冲突
mktemo 选项 格式filename.XXX X至少要出现三个
-d创建临时目录
-pDir 指明临时文件所存放目录位置
# mktemp test.XXX XXX生成随机字符
test.6Gk 执行结果
# mktemp /tmp/test.XXX
/tmp/test.631
# ll /tmp/test.631
-rw-------. 1 root root 0 8月23 14:28 /tmp/test.631
================================================================
安装复制文件
install命令:
install [选项] -tdirectory=目录来源 将源文件所有参数复制到指定目录
install [选项] -T来源 directory 将目标文件视为普通文件
install [选项] -ddirectory 创建空目录
-m mode 默认755
-o owner
-g group
举例:
# install -m 700 -o hacker -g hacker file1 file2 将file复制到file2 并且带有700的权限
rwx------.1 hacker hacker 257 8月23 20:50 file2 执行结果
# install -m 700 -d /testdir/installdir 创建空目录 给上700权限
# ll /testdir
总用量 0
drwx------. 2 root root 6 8月23 20:56 installdir 执行结果
=============================================================
bash如何展开命令行执行:
1 把命令行分成单个命令词
2 展开别名
3 展开大括号中的声明 ( {} )
4 展开波浪符声明( ~ )
5 命令替换$() 和 ``
6 再次把命令行分成命令词
7 展开文件通配(*、?、等等)
8 准备I/O重定向(< 、>)
9 运行命令
防止扩展:
单引号(')防止所有扩展
双引号(")也防止所有扩展但是一下情况例外
$ (美元符号)变量扩展
` (反引号) 命令替换
\ (反斜线) 禁止单个字符扩展
! (叹号) 历史命令替换
反斜线(\)会使随后的字符按愿意解释
bash的配置文件:按生效范围划分,存在两类
全局配置:
/etc/profile
/etc/profile.d/*.sh
/etc/bashrc
个人配置:
~/.bash_profile
~/.bashrc
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
profile为交互式登录的shell提供配置
全局:/etc/profile /etc/profile.d/*.sh
个人:~/.bash_profile
功能:用于定义环境变量 ; 运行命令或脚本
bashrc 为非交互式和交互式登录的shell提供配置
全局:/etc/bashrc
个人:~/.bashrc
功能:定义命令别名和函数 ; 定义本地变量
修改profile和bashrc文件后需要生效方法有两种
重新启动shell进程
. 或 source
bash 退出任务
保存在~/.bash_logout文件中(用户)
在退出登录shell时运行
可以用于:
创建自动备份
清楚临时文件
页:
[1]