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

相关推荐

  • MySQL架构

    1.MySQL整体逻辑架构 我们先下图看看MySQL整体逻辑架构(MySQL’s Logical Architecture)                       图1  第一层,即最上一层,所包含的服务并不是MySQL所独有的技术。…

    Linux干货 2015-04-13
  • 第三周作业

    1、列出1、列出当前系统上所有已经登录的用户的用户名,注意:同一个用户登录多次,则只显示一次即可。[root@localhost ~]# who | cut -d ‘ ‘ -f1 | sort -u 2、取出最后登录到当前系统的用户的相关信息。[root@localhost ~]# who | tail -1 | export &am…

    Linux干货 2017-12-16
  • LAMP编译安装

    所谓的LAMP是指构建在Linux之上经典的一种web架构方式,其中L指的是Linux,A指的是Apache的httpd,M指的是Mysql或MariaDB,P指的是PHP,这种组合方式的基本工作模式是有httpd负责接收和相应用户的请求,然后将需要处理的php页面交由后端的php应用程序处理,而php应用程序处理php页面时的数据存储在Mysql数据库中,…

    Linux干货 2016-12-19
  • 三剑客之Sed

    sed:stream editor(流编辑器) 工作特性:并不直接处理文本文件本身,处理机制为每当处理一个文件的时候,它会逐行读取,每次把一行读取到内存空间中去,而后在模式空间(pattern space)中完成编辑.并把编辑好的结果输出到屏幕上  功      能:数据替换、删除、增加、等,数据为关键字或者一整行, …

    Linux干货 2016-08-08
  • 删除boot,同时删除/etc/fstab,如何恢复

    删除boot,同时删除/etc/fstab,如何恢复 /etc/fstab文件丢失的时候,得光盘启动进入linux rescue下。 挂载好cdrom后,按ctrl+alt+del,然后按f2,进入bios,调整为cdrom启动,然后reboot,进入linux rescue急救模式/etc/fstab配置文件决定了linux系统在启动后如何加载各个分区,如…

    2017-09-23