当Shell遇到一个"$"符时(没有被引用或逃逸),它将认为其后为一变量。不论该变量是环境变量还


是用户自定义的 变量,在命令行中变量名要被变量值替换。例如命令:ls


$HOME将列出变量HOME对应目录下的文件。用户可以在命令行中的任何地方进行变量替换。包括命令名本身,例如:


    $dir=ls

    $$dir f*


  将列出以f开头的文件。


  现在详细的介绍下Bsh的变量。Bsh中有四类变量:用户定义的变量、位置变量(shell参数)、预定义变量及环境变量。


  用户定义的变量:


  用户定义的变量由字母和下划线组成,并且变量名的第一个字符不能为数字(0~9)。与其他UNIX名字一样,变量名是大小写敏感的。用户可以在命令行 上用"="给变量赋值,例如:


    $NAME=ice_walk


  给变量NAME赋值为ice_walk,在应用变量NAME的时候,在NAME前加"$"即可,前面已说,不再废话(别说我废话多,关键是没当过老 师)。可以用变量和其他字符组成新的字,例如:


    $SUN=sun

    $echo ${SUN}day


  在应用shell变量时候,可以在变量名字两边$后面加上{},以更加清楚的显示给shell,哪个是真正的变量,以实现字符串的合并等功能。

 

  结果显示:sunday(注意不能echo $SUNday,因为SUNday变量没定义,读者试下执行结果)

用户也可以在命令行上同时对多个变量赋值,赋值语句之间用空格分开:


    $X=x Y=y


    注意变量赋值是从右到左进行的


    $X=$Y Y=y

    X的值是y

    $X=z Y=$Z


    Y的值是空(变量未赋值时,shell不报错,而是赋值为空)


  用户可以使用"unset <变量>"命令清除给变量赋的值


用户使用变量时要在其前面加一"$"符,使变量名被变量值所替换。Bsh可以进行变量的条件替换,即只


有某种条件发生时才进行替换。替换条件放在一对大括 号{}中,如:


    ${variable: -value}

variable是一变量值,value是变量替换使用的默认值


    $echo Hello $UNAME

    结果显示:Hello

    $echo Hello ${UNAME: -there}

    结果显示:Hello there

    $echo $UNAME

    结果显示: (空)

    $UNAME=John

    $echo Hello ${UNAME: -there}

    结果显示:Hello John


  可以看出,变量替换时将使用命令行中定义的默认值,但变量的值并没有因此而改变。


另外一种替换的方法是不但使用默认值进行替换,而且将默认值赋给该变 量。其形式如下:


    ${variable:=value}


  该形式在变量替换后同时把值value符给变量variable。


    $echo Hello $UNAME

    结果显示:Hello

    $echo Hello ${UNAME:=there}

    结果显示:Hello there

    $echo $UNAME

    结果显示:there

    $UNAME=John

    $echo Hello ${UNAME:-there}

    结果显示:Hello John


  变量替换的值也可以是` `括起来的命令:


    $USERDIR={$Mydir: -`pwd`}


  第三种变量的替换方法是只有当变量已赋值时才用指定值替换形式:


    ${variable: +value}


    只有变量variable已赋值时,其值才用value替换,否则不进行任何替换,例如:


    $ERROPT=A

    $echo ${ERROPT: +"Error tracking is acitive"}

    结果显示:Error tracking is acitive

    $ERROPT=

    $echo ${ERROPT: +"Error tracking is acitive"}

    结果显示: (空)


  我们还可以使用错误检查的条件进行变量替换:


    ${variable:?message}


当变量variable已设置时,正常替换。否则消息message将送到标准错误输出(若此替换出现在shell程序中,那么该程序将终止)。 例如:


    $UNAME=

    $echo $ {UNAME:?"UNAME HAS NOT BEEN SET"}

    结果显示:UNAME HAS NOT BEEN SET


    $UNAME=Stephanie

    $echo $ {UNAME:?"UNAME HAS NOT BEEN SET"}


    结果显示:Stephanie

    当没有指定message时,shell将显示一条默认的消息,例如:


    $UNAME=

    $echo $ {UNAME:?}

    结果显示:sh:UNAME:parameter null or not set  







shell是解释型语言

    不象用C++/JAVA语言编程,不需要事先声明变量.

    用户可以使用同一个变量,"时而"存放字符,"时而"存放整数.

    字符串变量赋值很简单

logfile="/Users/mark/GPS.txt"        



    给变量赋值的注意事项

    abc=9 (bash/pdksh,不能在等号两侧留下空格 )


    set abc = 9 (csh /tcsh,正相反,必须在两侧留下空格)



    字符串变量赋值,其值一般都用引号   

            

    但也可以不用引号(前提是字符串不成句,不能有空格)

name=hello

echo $name


[root@mac-home macg]# ./re.test

hello

name=hello world     错,不允许有空格

echo $name

[root@mac-home macg]#./re.test

./re.test: line 2: world: command not found

name="hello world"   有空格,就必须用双引号

echo $name   

[root@mac-home macg]# ./re.test

hello world



   shell变量名中,不能有“中杠-",中杠会被识别为减号,必须改成"下杠_"

LAN-INT="eth0"

/sbin/iptables -A INPUT -i $LAN-INT -j ACCEPT

系统提示不认识LAN-INT=eth0

原因:中杠会被识别为减号

解决:改成下杠就好了

LAN_INT="eth0"

/sbin/iptables -A INPUT -i $LAN_INT -j ACCEPT



    变量引用

普通UNIX命令引用变量,要带$

sec=/home/macg

cd $sec

echo $r1

echo 双引号内引用变量,也要带$

echo "file $old is now called $new \c" 

变量也可直接执行,即变量值必须是一个命令字符串

ltest="ls -l"

$ltest

(read和赋值=和export,引用变量,不带$)

new="test"

read old

export PATH


   字符串变量在if条件里,最好用" "引起来                      

read timeofday

if [ $timeofday = “yes” ]

如果read输入的仅是回车,if语句看起来就是下面的样子:

if [ = “yes” ]

我们就会得到下面的错误信息:

[: =: unary operator expected

为了避免这样的问题,我们可以用双引号将变量括起来

当传递一个空串进行测试时:

if [ “$timeofday” = “yes” ]

if [ “” = “yes” ]    走else


     

   类似单引号,反斜线"\" 也能屏蔽所有特殊字符.但一次只能屏蔽一个字符.

echo "\$1 is $1"

echo "this is $2"

 # sh ttt

$1 is aa

this is bb

#echo “\$HOME = $PATH”   

# sh ttt

$HOME = /bin:/etc:/usr/bin



   ""   引号中间没有任何字符,表示回车

$ vi ttt.sh

echo -n "input:"

read tt

if [ "$tt" = "" ]   如果等于回车,则

then

echo haha

fi 

$ sh ttt.sh

input:      此处回车

haha


$ sh ttt.sh

input:x 


$ cat in.sh

while true

do

  echo -n "input:"

  read msg

  if [ "$msg" = "" ]

  then

continue

  else

echo "your input is $msg"

  fi  

$ sh in.sh

input:3

your input is 3

input:4

your input is 4

input:               回车走continue

input:

input:  



    local 变量 ——只支持函数内变量定义

test()

{

        local i


        for i in $* ; do

             echo "i is $i"

        done

test.sh: line 10: local: can only be used in a function


   删除变量unset,一般放在script结尾,释放空间

name="hello world"

echo $name

unset name


    养成良好的习惯,在shell程序结束,就unset 变量和函数

pathmunge () {

        if ! echo $PATH | /bin/egrep -q "(^|:)$1($|:)" ; then

           if [ "$2" = "after" ] ; then

              PATH=$PATH:$1

           else

              PATH=$1:$PATH

           fi

        fi

}

for i in /etc/profile.d/*.sh ; do

    if [ -r "$i" ]; then

        . $i

    fi

done


unset i

unset pathmunge


Shell 变量

定义变量时,变量名不加美元符号($,PHP语言中变量需要),如:

your_name="w3cschool.cc"

注意,变量名和等号之间不能有空格,这可能和你熟悉的所有编程语言都不一样。同时,变量名的命名须遵循如下规则:

  • 首个字符必须为字母(a-z,A-Z)。

  • 中间不能有空格,可以使用下划线(_)。

  • 不能使用标点符号。

  • 不能使用bash里的关键字(可用help命令查看保留关键字)。


除了显式地直接赋值,还可以用语句给变量赋值,如:

for file in `ls /etc`

以上语句将 /etc 下目录的文件名循环出来。


使用变量

使用一个定义过的变量,只要在变量名前面加美元符号即可,如:

your_name="qinjx"
echo $your_name
echo ${your_name}

变量名外面的花括号是可选的,加不加都行,加花括号是为了帮助解释器识别变量的边界,比如下面这种情况:

for skill in Ada Coffe Action Java do
    echo "I am good at ${skill}Script"
done

如果不给skill变量加花括号,写成echo "I am good at $skillScript",解释器就会把$skillScript当成一个变量(其值为空),代码执行结果就不是我们期望的样子了。

推荐给所有变量加上花括号,这是个好的编程习惯。


已定义的变量,可以被重新定义,如:

your_name="tom"
echo $your_name
your_name="alibaba"
echo $your_name

这样写是合法的,但注意,第二次赋值的时候不能写$your_name="alibaba",使用变量的时候才加美元符($)


Shell 字符串

字符串是shell编程中最常用最有用的数据类型(除了数字和字符串,也没啥其它类型好用了),字符串可以用单引号,也可以用双引号,也可以不用引号。单双引号的区别跟PHP类似。

单引号

str='this is a string'

单引号字符串的限制:

  • 单引号里的任何字符都会原样输出,单引号字符串中的变量是无效的;

  • 单引号字串中不能出现单引号(对单引号使用转义符后也不行)。

双引号

your_name='qinjx'
str="Hello, I know your are \"$your_name\"! \n"

双引号的优点:

  • 双引号里可以有变量

  • 双引号里可以出现转义字符

拼接字符串

your_name="qinjx"
greeting="hello, "$your_name" !"
greeting_1="hello, ${your_name} !"
echo $greeting $greeting_1

获取字符串长度

string="abcd"

提取子字符串

string="alibaba is a great company"

查找子字符串

string="alibaba is a great company"

Shell 数组

bash支持一维数组(不支持多维数组),并且没有限定数组的大小。

类似与C语言,数组元素的下标由0开始编号。获取数组中的元素要利用下标,下标可以是整数或算术表达式,其值应大于或等于0

定义数组

在Shell中,用括号来表示数组,数组元素用"空格"符号分割开。定义数组的一般形式为:

	数组名=(值1 值2 ... 值n)

例如:

array_name=(value0 value1 value2 value3)

或者

array_name=(
value0
value1
value2
value3
)

还可以单独定义数组的各个分量:

array_name[0]=value0
array_name[1]=value1
array_name[n]=valuen

可以不使用连续的下标,而且下标的范围没有限制。

读取数组

读取数组元素值的一般格式是:

    ${数组名[下标]}

例如:

valuen=${array_name[n]}

使用@符号可以获取数组中的所有元素,例如:

echo ${array_name[@]}

获取数组的长度

获取数组长度的方法与获取字符串长度的方法相同,例如:

# 取得数组元素的个数


# 或者



# 取得数组单个元素的长度

Shell 注释

以"#"开头的行就是注释,会被解释器忽略。

sh里没有多行注释,只能每一行加一个#号。只能像这样:

#--------------------------------------------
# 这是一个自动打ipa的脚本,基于webfrogs的ipa-build书写:
# https://github.com/webfrogs/xcode_shell/blob/master/ipa-build
# 功能:自动为etao ios app打包,产出物为14个渠道的ipa包
# 特色:全自动打包,不需要输入任何参数
#--------------------------------------------
##### 用户配置区 开始 #####
#
#
# 项目根目录,推荐将此脚本放在项目的根目录,这里就不用改了
# 应用名,确保和Xcode里Product下的target_name.app名字一致
#
##### 用户配置区 结束  #####

如果在开发过程中,遇到大段的代码需要临时注释起来,过一会儿又取消注释,怎么办呢?

每一行加个#符号太费力了,可以把这一段要注释的代码用一对花括号括起来,定义成一个函数,没有地方调用这个函数,这块代码就不会执行,达到了和注释一样的效果。