shell脚本编写-5

 

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

(0)
上一篇 2016-09-01 08:50
下一篇 2016-09-01 08:50

相关推荐

  • 利用keepalived实现nginx七层负载均衡器双主模型高可用集群

    nginx通过ngx_http_upstream_module可以实现七层http服务的负载均衡,同时还自带RS的健康监测功能,但是也有弊端,nginx负载均衡器如果发生故障则整个集群失效,此文通过keepalived构建nginx负载均衡器的高可用集群。 一、实验环境: 实验需要4台服务器: nginx负载均衡器1:192.168.11.100 nginx…

    2017-07-03
  • 初至马哥教育

    linux面授班30期

    2018-03-26
  • 网络N22期-第二周作业

    1、Linux上的文件管理类命令都有哪些,其常用的使用方法及其相关示例演示。 常用文件管理类命令有cp、mv、rm。 # cp命令:文件复制命令     cp [OPTION]… [-T] SOURCE DEST 单文件复制     cp [OPTION]… SOURC…

    Linux干货 2016-08-22
  • 第二周

    第二周 1. Linux上的文件管理类命令都有哪些,其常用的使用方法及其相关示例演示。 cp 复制文件和目录语法格式:cp [OPTION]… [-T] SOURCE DESTcp [OPTION]… SOURCE… DIRECTORYcp [OPTION]… -t DIRECTORY SOURCE&#8230…

    Linux干货 2017-07-21
  • VMware虚拟机三种连网原理简介

    一、前言         虚拟机(Virtual Machine),在计算机科学中的体系结构里,是指一种特殊的软件,可以在计算机平台和终端用户之间创建一种环境,而终端用户则是基于这个软件所创建的环境来操作系统软件。虚拟机根据它们运用和与直接机器的相关性分为两大类。系统虚拟机和程序虚拟机,系…

    Linux干货 2015-08-03
  • 马哥教育网络19期+第四周练习博客

    1、复制/etc/skel目录为/home/tuser1,要求/home/tuser1及其内部文件的属组和其它用户均没有任何访问权限。  cp /etc/skel /home/tuser1  chmod -R  700 /home/tuser1 2、编辑/etc/group文件,…

    Linux干货 2016-06-19