shell脚本编程 4

Shell脚本编程4

一、杂项知识整理

1、数组赋值方法:

declare -a files

files=(/var/log/*.log)通配所有文件也可

2、创建临时文件:

mktemp命令:创建的临时文件可避免冲突

mktemp  NAME.XXXX(x至少要出现三个)

-d 创建临时目录

–tmpdir=/DIR 指明临时文件所存放的目录位置

[root@localhost tmp.X2wtb]# mktemp -d /testdir/tmp.XXXXX
/testdir/tmp.yHrX9
[root@localhost tmp.X2wtb]# ll /testdir/tmp.*
/testdir/tmp.X2wtb:
总用量 0
/testdir/tmp.yHrX9:
总用量 0

3、install命令:安装复制文件:

-m  MODE 指定权限,默认为755

-o  OWNER 指定所有者

-g  GROUP 指定所属组

[root@localhost testdir]# install -m 770 -o root -g root -d tmpinstall
[root@localhost testdir]# ll tmpinstall/
总用量 0
[root@localhost testdir]# ll tmpinstall
总用量 0
[root@localhost testdir]# ll -d  tmpinstall
drwxrwx--- 2 root root 4096 8月  23 17:40 tmpinstall

4、bash如何展开命令行:

把命令行分成单个命令词

展开别名

展开大括号中的声明

展开波浪符声明

命令替换

再次把命令行分成命令词

展开文件通配

准备I/O重导向(<>

运行命令

5、bash的配置文件:按照生效范围划分,存在两类:

全局配置:/etc/profile、/etc/profile.d/*.sh/etc/bashrc

个人配置:~/.bash_profile    ~/.bashrc

shell登录的两种方式:

交互式登录:

直接通过终端输入账号密码登录

使用su – 切换用户

配置文件执行顺序:/etc/profile、/etc/profile.d/*.sh~/.bash_profile~/.bashrc/etc/bashrc

非交互式登录:

su 登录

图形界面下打开的终端

执行脚本

执行顺序:~/.bashrc、/etc/bashrc/etc/profile.d/*.sh

按照功能划分:profile类和bashrc类:

profile类:为交互式登录的shell提供配置

全局:/etc/profile、/etc/profile.d/*.sh

个人:~/.bash_profile

功能:用于定义环境变量;运行命令或脚本

bashrc类:为非交互式和交互式登录的shell提供配置

全局:/etc/bashrc

个人:~/.bashrc

功用:定义命令别名和函数;定义本地变量

bash退出任务:保存在~/.bash_logout文件中(用户)

在退出登录shell时运行;

用于创建自动备份或者清除临时文件等。

 

二、命令详解及事例

1、数组:存储多个元素的连续的内存空间,相当于多个变量的集合。

索引:编号从0开始,属于数值索引;

注意:索引可支持使用自定义的格式,而不仅是数值格式,即为关联索引,bash4.0之后开始支持。

Bash的数组支持稀疏格式(索引不连续)

声明数组:

declare  -a  ARRAY_NAME

Declare  -A  ARRAY_NAME:关联数组

数组元素的赋值:

一次只赋值一个元素

ARRAY_NAME[INDEX]=NALUE

weekdays[0]=sunday

weekdays[4]=thursday

一次赋值全部元素

ARRAY_NAME=(VAL1 VAL2 VAL3…)

只赋值特定元素

ARRAY_NAME=([0]=VAL1  [3]=VAL2…)

交互式赋值

命令:read -a ARRAY

引用数组元素:${ARRAY_NAME[INDEX]}

注意:省略[]表示引用下标为0的元素

数组的长度(数组中元素的个数):

${#ARRAY_NAME[*]}   ${#ARRAY_NAME[@]}

示例:生成10个随机数保存于数组中,并找出最大值和最小值:

#!/bin/bash
#array
#
declare -a RAND
declare -i MAX=0
declare -i MIN=32766
for I in {0..9};do
    RAND[$I]=$RANDOM
    echo ${RAND[$I]}
    [ "${RAND[$I]}" -ge "$MAX" ] && MAX=${RAND[$I]}
    [ "${RAND[$I]}" -le "$MIN" ] && MIN=${RAND[$I]}
done
    echo "MAX:$MAX"
    echo "MIN:$MIN"

写一个脚本,定义一个数组,数组中的元素是/var/log目录下所有以.log结尾的文件;

要统计其下标为偶数的文件中的行数之和:

#!/bin/bash
#array
#
declare -a FILE
declare -i LINE=0
FILE=(/var/log/*.log)
for i in $(seq 0 $[${#FILE[@]}-1]);do
    if [ $[$i%2] -eq 0 ];then
        let LINE+=$(wc -l ${FILE[$I]} | cut -d" " -f1)
    fi  
done
echo "Lines:$LINE"

2、数组数据处理:

引用数组中的元素:所有元素${ARRAY[@]},${ARRAY[*]}

数组切片:${ARRAY[@]:offset:number}

offset:要跳过的元素个数

number:要取出的元素个数

取偏移量之后的所有元素:${ARRAY:offset}

[root@localhost shelltest]# ing=(a b c d e)
[root@localhost shelltest]# echo ${ing}
a
[root@localhost shelltest]# echo ${ing[3]}
d
[root@localhost shelltest]# echo ${ing[*]:2}
c d e
[root@localhost shelltest]# echo ${ing[*]:2:1}
c

向数组中追加元素:

[root@localhost shelltest]# ing[${#ing[*]}]=f
[root@localhost shelltest]# echo ${ing[5]}
f

输出数组中的某单个元素:unset ARRAY[INDEX]

关联数组:declare -A ARRAY_NAME必须先声明

ARRAY_NAME=([idx_name1]=val1 [idx_name2]=val2…)

索引编号不是数字,自定义索引

[root@localhost shelltest]# declare -A jing
[root@localhost shelltest]# jing=([a]=aaa [b]=bbb [c]=ccc)
[root@localhost shelltest]# echo ${jing[c]}
ccc
[root@localhost shelltest]# echo ${jing[b]}
bbb
[root@localhost shelltest]# echo ${jing[a]}
aaa

3、字符串处理:字符串切片:

${#var} 返回字符串变量var的长度;

${var:offset} 返回字符串var中从第offset个字符后(不包含)的字符开始,到最后的部分,offset的取值在0${#var}-1之间(从bash4.2之后,允许为负值)。

${var:offset:number}:返回字符串变量var中从第offset个字符后(不包括第offset个字符)的字符开始,长度为number的部分。

${var:  -lengh}:取字符串的最右侧几个字符。冒号后必须有一空白字符,如果不是用空白字符分隔,没有意义。例如:

[root@localhost ~]# var=alasdjg21as4asd:asdg1:asd:asdf:asf
[root@localhost ~]# echo ${var:  -2}
sf
[root@localhost ~]# echo ${var: 5}
jg21as4asd:asdg1:asd:asdf:asf
[root@localhost ~]# echo ${var:5}
jg21as4asd:asdg1:asd:asdf:asf
[root@localhost ~]# echo ${var:5:12}
jg21as4asd:a

基于模式取子串:${var#*word} 其中word可以是指定的任意字符

功能:自左而右,查找var变量所存储的字符串中,第一次出现的word,删除字符串开头至第一次出现word字符之间的所有字符。

${var##*word} 同上,不同的是:删除的是字符串开头至最后一次由word指定的字符之间的所有内容。例如:

[root@localhost ~]# var=alasdjg21as4asd:asdg1:asd:asdf:asf
[root@localhost ~]# echo ${var#*:}
asdg1:asd:asdf:asf
[root@localhost ~]# echo ${var##*:}
asf

${var%word*} 其中word可以是指定的任意字符:

功能,自右往左,查找var变量存储的字符串中,第一次出现的word,删除字符串最后一个字符向左至第一次出现word字符之间的所有字符;例如:

[root@localhost ~]# var=alasdjg21as4asd:asdg1:asd:asdf:asf
[root@localhost ~]# echo ${var%:*}
alasdjg21as4asd:asdg1:asd:asdf
[root@localhost ~]# echo ${var%%:*}
alasdjg21as4asd

查找替换:

${var/pattern/substi} 查找var所表示的字符串中,第一次被pattern所匹配到的字符串,以substi替换之。

${var//pattern/subti} 查找var所表示的字符串中,所有能被pattern所匹配到的字符串,以substi替换之

${var/#pattern/substi} 查找var所表示的字符串中,行首被pattern所匹配到的字符串,以substi替换之。

${var/%pattern/substi} 查找var所表示的字符串中,行尾被pattern所匹配到的字符串,以substi替换之。

[root@localhost ~]# echo ${var/:/@@}
alasdjg21as4asd@@asdg1:asd:asdf:asf
[root@localhost ~]# echo ${var//:/@@}
alasdjg21as4asd@@asdg1@@asd@@asdf@@asf
[root@localhost ~]# echo ${var/#@/vvv}
alasdjg21as4asd:asdg1:asd:asdf:asf
[root@localhost ~]# echo ${var/#:/vvv}
alasdjg21as4asd:asdg1:asd:asdf:asf
[root@localhost ~]# echo ${var/#/vvv}
vvvalasdjg21as4asd:asdg1:asd:asdf:asf
[root@localhost ~]# echo ${var/%/AAA}
alasdjg21as4asd:asdg1:asd:asdf:asfAAA
[root@localhost ~]# echo ${var/%asf/AAA}
alasdjg21as4asd:asdg1:asd:asdf:AAA

查找并删除:

${var/pattern} 查找var所表示的字符串中,删除第一次被pattern所匹配到的字符串

${var//pattern} 删除所有匹配到的字符串

${var/#pattern} 删除行首被匹配到的字符串

${var/%pattern} 删除行尾被匹配到的字符串

${var^^} 把var中的所有小写字母转换为大写

${var,,} 把var中的所有大写字母转换为小写

例如:

[root@localhost ~]# echo $var
alasdjg21as4asd:asdg1:asd:asdf:asf
[root@localhost ~]# echo ${var/%asf/AAA}
alasdjg21as4asd:asdg1:asd:asdf:AAA
[root@localhost ~]# echo ${var//as}
aldjg214d:dg1:d:df:f
[root@localhost ~]# echo ${var^^}
ALASDJG21AS4ASD:ASDG1:ASD:ASDF:ASF
[root@localhost ~]# echo ${a}
root:x:0:0:root:/root:/bin/bash
[root@localhost ~]# echo ${a/#*x}
:0:0:root:/root:/bin/bash

pattern皆支持通配符,处理方式基于类似pattern space,不直接改变字符串。

变量赋值:

${var:-value} 如果为空或未设置,那么返回value;否则返回var值。

${var:+value} 如果var不空则返回value

${var:=value} 如果var为空或未设置,那么返回value,并将value赋值给var;否则则返回var的值;

${var:?error_info} 如果var为空或未设置,那么返回error_info;否则分会var的值。

[root@localhost ~]# echo ${var:-AAA}
alasdjg21as4asd:asdg1:asd:asdf:asf
[root@localhost ~]# echo ${var:+AAA}
AAA
[root@localhost ~]# echo ${var:=AAA}
alasdjg21as4asd:asdg1:asd:asdf:asf
[root@localhost ~]# echo ${aaa:=AAA}
AAA
[root@localhost ~]# echo $aaa
AAA
[root@localhost ~]# echo ${va:?abc}
-bash: va: abc
[root@localhost ~]# echo ${var:?abc}
alasdjg21as4asd:asdg1:asd:asdf:asf
[root@localhost ~]# echo ${var:-AAA}
alasdjg21as4asd:asdg1:asd:asdf:asf
[root@localhost ~]# echo ${var:+AAA}
AAA
[root@localhost ~]# echo ${var:=AAA}
alasdjg21as4asd:asdg1:asd:asdf:asf
[root@localhost ~]# echo ${aaa:=AAA}
AAA
[root@localhost ~]# echo $aaa
AAA
[root@localhost ~]# echo ${va:?abc}
-bash: va: abc
[root@localhost ~]# echo ${var:?abc}
alasdjg21as4asd:asdg1:asd:asdf:asf
[root@localhost ~]# 
[root@localhost ~]#

aaa赋值:若aaaa存在则赋aaaa的值,不存在则赋值123456

[root@localhost ~]# aaa=${aaaa:=123456}
[root@localhost ~]# echo $aaa
123456

为脚本程序使用配置文件,实现变量赋值:

定义文本文件,每行定义NAME=VALUE;在脚本中source此文件即可。

4、declare命令:

-r 将变量设置为只读属性

-i 将变量定义为整型数

-a 将变量定义为数组

-f 显示此脚本前定义过的所有函数名及其内容

-F 仅显示此脚本前定义过的所有函数名

-x 将变量声明为环境变量

-l 将变量值转为小写字母

-u 将变量转为大写字母

[root@localhost ~]# declare -u a=aaa
[root@localhost ~]# echo $a
AAA

间接变量引用:如果第一个变量的值是第二个变量的名字,从第一个变量用用第二个变量的值就称为简介变量引用。bash中提供了两种格式实现间接变量引用:

eval tempvar=\$$variable1

eval命令:eval命令将会首先扫描命令行进行所有的置换,然后再执行该命令。该命令适用于那些一次扫描无法实现其功能的变量。该命令对变量进行两次扫描。

tempvar=${!variable1}

示例:

[root@localhost ~]# n=name
[root@localhost ~]# name=wang
[root@localhost ~]# NAME=${!n}
[root@localhost ~]# echo $NAME
wang
[root@localhost ~]# eval NAME1=\$$n
[root@localhost ~]# echo $NAME1
wang

eval命令示例:

[root@localhost ~]# abc=pwd
  [root@localhost ~]# eval $abc
/root


三、课后练习

1、生成10个随机数,采用冒泡算法进行升序或降序排序。

#!/bin/bash
#sort
#
read -p "ten number: " NUM 
if echo $NUM | grep [[:alpha:]] ;then
    echo "input right number!"
    exit 1
fi
declare -a num=($NUM)
declare -i min=0
echo ${num[@]}
 
for I in `seq 0 $[${#num[@]}-1]`;do
    for J in `seq 1 $[${#num[@]}-1]`;do
        if [ ${num[$J]} -ge ${num[$[$J-1]]} ];then
            min=${num[$[$J-1]]}
            num[$J-1]=${num[$J]}
            num[$J]=$min
        fi
    done
done
 
echo ${num[@]}

原创文章,作者:SilencePavilion,如若转载,请注明出处:http://www.178linux.com/39295

(0)
SilencePavilionSilencePavilion
上一篇 2016-08-24
下一篇 2016-08-24

相关推荐

  • linux操作系统rpm软件包管理

    软件包管理 软件包运行环境: API:Application Programming Interface     使用标准:POSIX:     源代码运行步骤:预处理(如处理注释)—>编译成汇编代码—>链接其他库文件 ABI:applicatio…

    2016-08-22
  • 文本处理工具sed、vim

    文本处理工具sed、vim 一、sed的使用 1、sed的概念 sed简单地说sed是一种行处理工具。 sed 是一种流性的行编辑器,它一次处理一行内容。处理时,把当前处理的行存储在临时缓冲区中,称为“模式空间”(pattern space ),接着用sed 命令处理缓冲区中的内容,处理完成后,把缓冲区的内容送往屏幕。接着处理下一行,这样不断重复,直到文件末…

    Linux干货 2016-08-11
  • find命令归纳

    首先,find可以实现以下方式查找 文件名:-name -iname -regex 文件类型:-type TYPE:(f,d,l,s,b,c,p) 文件大小:-size [+|-]#UNIT #UNIT(k,M,G)范围(由小到大):[0,#-1](#-1,#](+#,oo) 时间戳:-{a|m|c}time -{a|m|c}min  范围(现在到…

    Linux干货 2016-04-19
  • shell脚本总结

    shell进阶:列表生成方式:列表生成方式:(1) 直接给出列表 以空白为间隔(2) 整数列表:(a) {start..end}(b) $(seq [start [step]] end)(3) 返回列表的命令$(COMMAND)(4) 使用glob,如:.sh(5) 变量引用;$@, $while read line(用法)(用于遍历文件,进行处理…

    Linux干货 2017-07-10
  • 马哥教育网络班22期+第一周课程练习

    1、描述计算机的组成及其功能。 计算机由硬件系统、操作系统和应用软件所组成,没有安装任何软件的计算机称为裸机。 1.1、硬件系统由运算器、控制器、存储器、输入设备和输出设备五大部分组成,遵循冯、诺依曼原理。 控制器:读取分析指令,向其它部分发出控制信号,保证计算机按照预先规定的目标和步骤有条不紊地进行操作及处理。  运算器:对数据进行各种运算,例如…

    Linux干货 2016-08-22
  • shell编程2

    组合测试条件  第一种方式:        COMMAND1 && COMMAND2 并且        COMMAND1 || COMMAND2 或者        ! COMMAND 非   &nbs…

    Linux干货 2016-08-21