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)
1861276386318612763863
上一篇 2016-09-01
下一篇 2016-09-01

相关推荐

  • 马哥网络教育班21期+第一周博客作业

    1、描述计算机的组成及其功能。   硬件:     CPU(运算器、控制器、一缓存(L1)、二级缓存(L2)):       运算器主要用来进行算数运算以及逻辑运算。       控制器主要是用来协调       一级缓存和二级缓存主要是用…

    Linux干货 2016-07-12
  • DNS

    简介     DNS(Domain Name System,域名系统),因特网上作为域名和IP地址相互映射的一个分布式数据库,能够使用户更方便的访问互联网,而不用去记住能够被机器直接读取的IP数串。通过主机名,最终得到该主机名对应的IP地址的过程叫做域名解析(或主机名解析)。DNS协议运行在UDP协议之上,使用端口号5…

    Linux干货 2016-11-15
  • Linux-第一周作业

    1.计算机组成及其功能 计算机主要分为三部分: A、输入单元:包括键盘、鼠标、扫描仪、手写板、触摸屏等。 B、中央处理器(CPU):含有算术逻辑、控制、记忆等单元。 C、输出单元:例如屏幕、打印机等。 我们通过输入设备(如鼠标与键盘)来将一些数据输入到PC里面,然后再由PC的功能处理成为图表或文章等信息后,将结果传输到输出设备,如屏幕或打印机上面,这是计算机…

    Linux干货 2016-07-07
  • Redis数据库安全手册

    Redis是一个高性能的key-value数据库,这两年可谓火的不行。而Redis的流行也带来一系列安全问题,不少攻击者都通过Redis发起攻击。本文将讲解这方面的内容,包括Redis提供的访问控制和代码安全问题,以及可以由恶意输入和其他类似的手段触发的攻击。 Redis通用安全模块 Redis被设计成只能由可信环境的可信机器访问。这意味着将它直接暴露在互联…

    2015-03-12
  • lvm简要及基本操作

        LVM( Logical Volume Manage,逻辑 逻辑卷管理)LVM将一个或多个硬盘的分区在逻辑上集合,相当于一个大硬盘来使用,当硬盘的空间不够使用的时候,可以继续将其它的硬盘的分区加入其中,这样可以实现磁盘空间的动态管理,相对于普通的磁盘分区有很大的灵活性。  &nbs…

    Linux干货 2016-05-23
  • Mariadb数据库复制系列(四):复制过滤器的实现

       实验四:复制过滤器的实现 复制过滤器的作用是让从节点仅复制主节点的指定的库,或指定数据库的指定表,而不是全部复制。其配置方式有两种,一种是在主节点上配置,通过定义主节点只记录部分库或表产生的二进制日志时间来实现;一种是通过从节点上配置,当复制了主节点的所有的二进制日志数据后,在SQL线程执行时,只执行指定的库或表相关的二进制日志时间;一般采取在从节点上…

    Linux干货 2016-11-24