declare in shell[转]

declare peset选项 -r 只读 1 declare -r var1 (declare -r var1与readonly var1是完全一样的) 这和C语言中的const关键字一样, 都用来指定变量为只读. 如果你尝试修改一个只读变量的值, 那么会产生错误信息. -i 整型 1 declare -i number 2 # 脚本将会把变量"number"按照整型进行处理. 3 4 number=3 5 echo "Number = $number" # Number = 3 6 7 number=three 8 echo "Number = $number" # Number = 0 9 # 脚本尝试把字符串"three"作为整数来求值(译者注: 当然会失败, 所以出现值为0). 如果把一个变量指定为整型的话, 那么即使没有expr或者let命令, 也允许使用特定的算术运算. 1 n=6/3 2 echo "n = $n" # n = 6/3 3 4 declare -i n 5 n=6/3 6 echo "n = $n" # n = 2 -a 数组 1 declare -a indices 变量indices将被视为数组. -f 函数 1 declare -f 如果在脚本中使用declare -f, 而不加任何参数的话, 那么将会列出这个脚本之前定义的所有函数. 1 declare -f function_name 如果在脚本中使用declare -f function_name这种形式的话, 将只会列出这个函数的名字. -x export 1 declare -x var3 这句将会声明一个变量, 并作为这个脚本的环境变量被导出. -x var=$value 1 declare -x var3=373 declare命令允许在声明变量类型的同时给变量赋值. 转自:http://blog.chinaunix.net/u/122/showart_490989.html

shell数组学习[技术]

Bash中还可以使用数组变量,其赋值有两种:

(1) name = (value1 … valuen) 此时下标从0开始
(2) name[index] = value

数组下标的范围没有任何限制,同时也不必使用连续的分量.

[root@linux01 ~]# A=(a b c def)

==================================================

//取全部元素
[root@linux01 ~]# echo ${A[@]} 或者 echo ${A[*]}
a b c def

=================================================

//取第一个元素
[root@linux01 ~]# echo ${A[0]}
a

=================================================

//取得数组元素的个数
[root@linux01 ~]# echo ${#A[@]} 或者 echo ${#A[*]}
4

//取得元素3的长度,因为下标从0开始,所以元素3就是def
[root@linux01 ~]# echo ${#A[3]}
3

==================================================

[root@linux01 ~]# A[3]=wenzizone.cn //将第三个元素重新赋值
[root@linux01 ~]# echo ${A[@]}
a b c wenzizone.cn

==================================================
//清除变量
[root@linux01 ~]# unset A
[root@linux01 ~]# echo ${A[@]}
<没有输出>
==================================================

//清空变量,即将值变为空
[root@linux01 ~]# A=
[root@linux01 ~]# echo ${A[@]}

<没有输出>
==================================================

[root@linux01 ~]# A=B
[root@linux01 ~]# B=C
[root@linux01 ~]# unset $A 事实上所取消的变量是 B 而不是 A
[root@linux01 ~]# echo $A
B
[root@linux01 ~]# echo $B

<没有输出>
=======================示例================================
#! /bin/bash
cat 1.txt|while read line
do
        myarray=($line)
        echo ${#myarray[@]}
        for ((i=0;i<=${#myarray[@]};i++))
        do
                echo ${myarray[$i]}
        done
done

1.txt的内容是
# cat 1.txt
/usr/local/nagios abc 192.168.11.1 192.168.11.15
/usr/local/nagios def 192.168.12.1 192.168.12.15

程序运行结果

# ./tst.sh 
4
/usr/local/nagios
abc
192.168.11.1
192.168.11.15

4
/usr/local/nagios
def
192.168.12.1
192.168.12.15

Bash的字符串匹配共有6种形式

Bash的字符串匹配共有6种形式:

1. ${variable#pattern}
如果pattern匹配variable的开始部分,从variable的开始处删除字符直到第一个匹配的位置,包括匹配部分,返回剩余部分。

2. ${variable##pattern}
如果pattern匹配variable的开始部分,从variable的开始处删除字符直到最后一个匹配的位置,包括匹配部分,返回剩余部分。

3. ${variable%pattern}
如果pattern匹配variable的结尾部分,从variable的结尾处删除字符直到第一个匹配的位置,包括匹配部分,返回剩余部分。

4. ${variable%%pattern}
如果pattern匹配variable的结尾部分,从variable的结尾处删除字符直到最后一个匹配的位置,包括匹配部分,返回剩余部分。

5. ${variable/pattern/string}
6. ${variable//pattern/string}
最后这两种用法用于匹配替换。因为我没用到,先不说了。(busybox 1.0.1 不支持最后这两种语法。)

举例:
str=tftp://hostname.com/onepath/anotherpath

echo ${str#*/}
输出:/hostname.com/onepath/anotherpath

echo ${str##*/}
输出:anotherpath

echo ${str%/*}
输出:tftp://hostname.com/onepath

echo ${str%%/*}
输出:tftp:/

——————–
下面说一说具体怎么用。

假设在某个系统中没有sed和awk, 只有grep,tr和cut。
如果要取得网卡的mac地址,可以:

mac_addr=$(ifconfig eth0 | grep HWaddr | cut -d’ ‘ -f11 | tr -d : )
echo $mac_addr
输出:0150BF9886BF

后面再说为什么我要把冒号去掉。这种方式在cut时要经过实验才知道我们要的是第11个field。现在换另一种方式:

mac_addr=$(ifconfig eth0 | grep HWaddr | tr -d ‘ :’)
echo $mac_addr
输出:eth0LinkencapEthernetHWaddr0150BF9886BF
然后:
mac_addr=${mac_addr#*HWaddr}
echo $mac_addr
输出:0150BF9886BF

现在要求写一个程序,接受命令行给定一个网址去下载一个文件,要求根据网址的协议的不同,采用不同的程序下载。如果给定的网址以.xml结尾,则认为要下载的文件已经在给定的网址中指定,否则要下载的文件名为本机的mac地址加.xml扩展名,不包括mac中的冒号。

例如给定 tftp://host/file.xml,则要用tftp命令下载host上的file.xml文件。
如果给定 http://host/path,则要用wget命令下载host/path上的0150BF9886BF.xml文件。

先取网址的协议,采用从右向左最大匹配”://”:
url=$1
proto=${url%%://*}

再判断文件名是否已经给定, 采用从左向右最大匹配”.xml”:
[ -z “${url##*.xml}” ] || url=$url/$mac_addr.xml

如果 $proto = “http” 或者 “ftp”
则执行
wget $url -O local_file

如果 $proto = “tftp”,这个有点麻烦,因为tftp的用法是:

tftp -g -r remote_file -l local_file host

所以还要把remote_file和host从url中提取出来。

先把url中的tftp://去掉:
tmp=${url#*://}

再从右向左最大匹配”/”得到host:
host=${tmp%%/*}

再从左向右最小匹配”/”得到路径和文件名:
remote_file=${tmp#*/}