shell编程之数组及环境变量

数组

变量:存储单个元素的内存空间

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

索引:编号从0

声明数组:

declare -a ARRAY_NAME

declare -A ARRAY_NAME:关联数组

1.declare  -a menu

 menu[0]=beef  menu[1]=chicken menu[2]=pork  //一次只赋一个元素

 echo ${menu[*]} 输出beef chicken pork

2. weekdays=("Sunday" "Monday" "Tuesday" "Wednesday" "Thursday " "Friday" "Saturday")  //一次赋全部元素

3. playername=([0]="Alice" [3]="Tom") //只赋值特定元素

4. read -a array  //交互式数组值对赋值


引用数组元素:${ARRAY_NAME[INDEX]},省略[INDEX] 表示引用下标为0

数组的长度( 数组中元素的个数): ${#ARRAY_NAME[*]}   ${#ARRAY_NAME[@]}

files=(/boot/*)

echo ${files[*]}  //显示boot目录下所有文件

/boot/f1 /boot/grub /boot/grub2 /boot/initramfs-0-rescue-abfbe0705f7e46b99c5ac54db420992c.img /boot/initrd-plymouth.img

/boot/vmlinuz-0-rescue-abfbe0705f7e46b99c5ac54db420992c

 

fff=(*)

echo ${fff[*]}   //显示当前目录列表

 

ff=({1,2}{a,b})

echo ${ff[*]}  输出 1a 1b 2a 2b

 

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

blob.png

 

写一个脚本,定义一个数组,数组中的元素是/var/log 目录下所有以.log 结尾的文 件;要统计其下标为偶数的文件中的行数之和

blob.png

 

数组数据处理

引用数组中的元素:

所有元素:${ARRAY[@]}, ${ARRAY[*]}

echo ${menu[*]} 显示 beef chicken pork fish

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

offset:要跳过的元素个数    number:要取出的元素个数

echo ${menu[@]:1:2} 显示 chicken pork"

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

echo ${menu[@]:1} 显示 chicken pork fish"

删除数组中的某元素:导致稀疏格式 unset ARRAY[INDEX]

echo ${weekdays[*]} 显示 Sunday Monday Tuesday Wednesday Thursday Friday Saturday

unset weekdays[3] ,echo ${weekdays[3]} 显示为空,weekdays的元素中已没有Wednesday此元素

向数组中追加元素:ARRAY[${#ARRAY[*]}]

hobbies元素有swimming skiing surfing ,hobbies[${#hobbies[*]}]=jumpping

echo ${hobbies[*]} 显示 swimming skiing surfing jumpping

      

字符串处理

字符串切片:   str="hello world"

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

echo ${#str}输出11

${var:offset}: 返回字符串变量var 中从第offset 个字符 后(不包括第offset 个字符)的字符开始,到最后的部分,offset 的取值在0 到 到 ${#var}-1 之间

echo ${str:2}输出llo world

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

echo ${str:3:5}输出lo wo

${var: -lengh} :取字符串的最右侧几个字符,注意:冒号后必须有一空白字符

echo ${str:-2}输出 ld

       

字符串处理

基于模式取子串:   s='root:x:0:0:root,yanfa30,110,10086:/root:/bin/bash'

${var#*word}:其中word 可以是指定的任意字符 功能:自左而右,查找var 变量所存储的字符串中,第一 次出现的word,删除字符串开头至第一次出现word 字符之间的所有字符  

echo ${s#*root}  输出 :x:0:0:root,yanfa30,110,10086:/root:/bin/bash

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

echo ${s##*root} 输出 :/bin/bash

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=http://www.magedu.com:80

echo ${url##*:} 输出 80

echo ${url%%:*} 输出 http

 

查找替换:  s='root:x:0:0:root,yanfa30,110,10086:/root:/bin/bash'

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

echo ${s/0/zero} 输出 root:x:zero:0:root,yanfa30,110,10086:/root:/bin/bash

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

echo ${s//root/admin} 输出 admin:x:0:0:admin,yanfa30,110,10086:/admin:/bin/bash

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

echo ${s/#root/admin} 输出 admin:x:0:0:root,yanfa30,110,10086:/root:/bin/bash

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

echo ${s/%bash/nologin} 输出 root:x:0:0:root,yanfa30,110,10086:/root:/bin/nologin

 

查找并删除:  str="release:centos5 centos6 centos7"

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

echo ${str/centos} 输出 release:5 centos6 centos7

${var//pattern}:所有

echo ${str//centos} 输出 5 6 7

${var/#pattern}:首行

echo ${str/#release} 输出 :centos5 centos6 centos7

${var/%pattern}:行尾

echo ${str/%7} 输出  release:centos5 centos6 centos

 

字符大小写转换:  title='hello world'

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

echo ${title^^} 输出 HELLO WORLD

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

Title='HELLO WORLD'    echo ${Title,,} 输出 hello world

 

变量赋值

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

name= ;echo ${name:-alice} 输出 alice     name=bob;echo ${name:-alice} 输出bob

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

name=bob;echo ${name:+alice} 输出 alice     name= ;echo ${name:+alice} 输出空

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

name= ;echo ${name:=alice};echo $name 输出alice,name的值为alice       

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

echo ${promt:?warning} 输出 -bash: promt: warning   

promt='Are u sure?';echo ${promt:?warning} 输出 Are u sure?

 

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

(1)定义文本文件,每行定义“name=value”

(2)在脚本中source

Shell 变量一般是无类型的,但是bash Shell 提供了declare和 typeset 两个命令用于指定变量的类型,两个命令是完全等价

declare [ 选项]  变量名

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

-i  将变量定义为整型数

-a  将变量定义为数组

-A  将变量定义为关联数组

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

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

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

-l 将变量值转为小写字母 declare –l var=UPPER

declare -l b="HALT";echo $b 输出 halt  

-u 母 将变量值转为大写字母 declare –u var=lower

declare -u a="hello world";echo $a 输出 HELLO WORLD

 

如果第一个变量的值是第二个变量的名字,从第一个变量引 用第二个变量的值就称为间接变量引用

variable1=variable2

variable2=value

variable1 的值是variable2 ,而variable2 又是变量名,

variable2 的值为value ,间接变量引用是指通过variable1获得变量值value

 

间接变量引用   u=username;username=root

格式一:tempvar=${!variable1}

u1=${!u};echo $u1 输出 root

格式一:eval tempvar=\$$variable1

eval u2=\$$u;echo $u2 输出 root

 

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

变量。该命令对变量进行两次扫描

CMD=hostname   eval $CMD 输出 localhost.localdomain

 

 

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

mktemp [OPTION]… [TEMPLATE]

TEMPLATE: filename.XXX ,X至少3个

OPTION: 

-d:  创建临时目录

-p DIR 或–tmpdir=DIR :指明临时文件所存放目录

 

tmpname=`mktemp tmp.XXX`    echo $tmpname 显示生成的临时文件 tmp.GXA

mktemp -p /testdir tmp.XXX  显示生成的临时文件 /testdir/tmp.6M9

mktemp -d new.XXX 显示生成的临时目录 new.7dc

 

安装复制文件

install 命令:

   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

install -m 666 -o hwq forexample.sh /testdir/examplefor.sh

install -m 600  -d /testdir/installdir    创建空目录installdir

install -m 755 /etc/passwd -t /testdir/installdir  将passwd文件复制到installdir目录下


bash展开命令行

把命令行分成单个命令词

展开别名

展开大括号种的声明({}) )

展开波浪符声明(~) )

命令替换$() 和 “) )

再次把命令行分成命令词

展开文件通配(* 、? 、[abc] 等等)

准备I/0 重导向(< 、>) )

 

反斜线(\ )会使随后的字符按原意解释

$ echo Your cost: \$5.00  输出 Your cost: $5.00

加引号来防止扩展

单引号(')防止所有扩展

双引号(")也防止所有扩展,但是以下情况例外:

$ (美元符号) - 变量扩展

` (反引号) - 命令替换

 \ (反斜线) - 禁止单个字符扩展

 ! (叹号) - 历史命令替换

 

bash的配置文件

按生效范围划分,存在两类:

全局配置:

/etc/profile

/etc/profile.d/*.sh

/etc/bashrc

个人配置:

~/.bash_profile

~/.bashrc

 

交互式登录:

  (1) 直接通过终端输入账号密码登录;

  (2) 使用“su – UserName”  切换的用户

  执行顺序:

  /etc/profile –> /etc/profile.d/*.sh –> ~/.bash_profile –> ~/.bashrc –> /etc/bashrc

/etc/profile中HISTSIZE=1000

~/.bash_profile 中HISTSIZE=2000  source使之生效后 echo $HISTSIZE 显示2000

非交互式登录:

  (1)su UserName

  (2) 图形界面下打开的终端

  (3) 执行脚本

  执行顺序:

  ~/.bashrc –> /etc/bashrc –> /etc/profile.d/*.sh

 

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

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

  个人:~/.bash_profile

  功用:

  (1)  用于定义环境变量

  (2)  运行命令或脚本

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

  全局:/etc/bashrc

  个人:~/.bashrc

  功用:

  (1)定义命令别名和函数

  (2)定义本地变量

 

修改profile 和bashrc 文件后需生效

1 重新启动shell 进程

2 或source

 

cat /etc/profile.d/vim.sh 

blob.png

su – wang 切到wang用户,alias可以看到alias vi='vim'

 

bash退出任务

保存在~/.bash_logout 文件中(用户)

在退出登录shell 时运行

适用于 创建自动备份 /清除临时文件

# ~/.bash_logout

rm -rf /root/tmp.gYZ/*

 

实战

1.让用户(管理员或所有用户)的PATH 环境变量的值多出 一个路径,例如:/usr/local/apache2/bin

   vim ~/.bash_profile

   PATH=.:$PATH:$HOME/bin:/usr/local/apache2/bin

2.用户wang 登录时自动启用别名rm=‘rm –i’

   [wang@localhost ~]$ vim .bashrc   

   alias rm='rm -i' 

3.用户登录时,显示红色字体的警示提醒信息  "hi,dangerous"

   vim /etc/profile.d/log.sh

   echo -e "\033[31mdangerous\033[0m"

   blob.png

 

 

输入若干个数值存入数组中,采用冒泡算法进行升序或降序排序

#!/bin/bash
echo -ne "Input several numbers: "
read -a num
for ((i=0;i<=$[${#num[*]}-1];i++))
do
        echo -ne "${num[i]} "
done
echo
 
 
for ((j=$[${#num[@]}-1];j>0;j--))
do
        for ((k=$[$j-1];k>=0;k--))
        do
                if [ ${num[j]} -lt ${num[k]} ]
                then
                        a=${num[j]}
                        num[j]=${num[k]}
                        num[k]=$a
                fi
        done
done
echo "正序排列: "
 
for ((i=0;i<=$[${#num[@]}-1];i++))
do
        echo -ne "${num[i]} "
done
echo
 
 
for ((j=$[${#num[@]}-1];j>0;j--))
do
        for ((k=$[$j-1];k>=0;k--))
        do
                if [ ${num[j]} -gt ${num[k]} ]
                then
                        a=${num[j]}
                        num[j]=${num[k]}
                        num[k]=$a
                fi
        done
done
echo    "逆序排列:"
 
for ((l=0;l<=$[${#num[@]}-1];l++))
do
        echo -ne "${num[l]} "
done
echo

blob.png

 

 

 

 

 

 

 

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

(0)
victorycommandervictorycommander
上一篇 2016-08-29
下一篇 2016-08-29

相关推荐

  • Linus Torvalds 语录 Top 10

    下面是Linux的创始人Linus Torvalds的一些言论,这是我个人认为最有意思的10句。如果你想看更多的Linus Torvalds说过的话,你可以看看他在维基百科上的词条:Linux Torvalds。我们在下面给出中英文对照,希望你能喜欢。 “Really, I’m not out to dest…

    Linux干货 2015-04-03
  • OpenStack Icehouse私有云实战部署

    前言 相信你一定对“云主机”一词并不陌生吧,通过在Web页面选择所需主机配置,即可快速定制一台属于自己的虚拟主机,并实现登陆操作,大大节省了物理资源。但这一过程是如何实现的呢?本文带来OpenStack Icehouse私有云实战部署。 OpenStack 简介 OpenStack是由网络主机服务商Rackspace和美国宇航局联合推出的一个开源项目,Ope…

    Linux干货 2015-07-29
  • Sed简介

    Sed简介    一、简介   sed全称是:Stream EDitor。sed命令的功能同awk类似,差别在于,sed简单,对列处理的功能要差一些,awk的功能复杂,对列处理的功能比较强大。  sed 是一种在线编辑器,它一次处理一行内容。处理时,把当前处理的行存储在临时缓冲区中,称为“模式空间”(pattern s…

    Linux干货 2015-05-11
  • 创建CA 和申请证书

    创建CA 和申请证书1生成私有CA的私钥:(umask 066;openssl genrsa -out /etc/pki/CA/private/cakey.pem 2048)注:CA的私钥文件必须放在/etc/pki/CA/private/cakey.pem 这个路径里2创建序列号 ,数据库文件touch  /etc/pki/CA/index.tx…

    Linux干货 2017-07-17
  • 第五周:grep和find的使用练习

    1、显示当前系统上root、fedora或user1用户的默认shell; grep -E "^(root|fedora|user1)" /etc/passwd | cut -d: -f7 grep -E "^root|fedora|use…

    Linux干货 2016-10-25
  • N21天天第十周课程练习

    1、请详细描述CentOS系统的启动流程(详细到每个过程系统做了哪些事情) CentOS主机按以下顺序启动 1、POST加电自检 2、BIOS读取CMOS中的BIOS设置的参数来识别基础硬件,寻找到启动设备 3、MBR 1)读取启动设备MBR中前446字节的bootloader 2)读取MBR后的扇区用来识别grub以及内核kernel所在的区域 3)启动g…

    Linux干货 2016-09-26