(6)結(jié)構(gòu)化編程定義函數(shù)
同其他高級(jí)語(yǔ)言一樣shell也提供了函數(shù)功能函數(shù)通常也稱之為子過(guò)
程(subroutine)
,
其定義格式如下
funcname()
{
command
...
command; #分號(hào)
}
定義函數(shù)之后可以在shell中對(duì)此函數(shù)進(jìn)行調(diào)用使用函數(shù)定義可以將
一個(gè)復(fù)雜的程序
分
為多個(gè)可管理的程序段如下所示
# start program
setup ()
{ command list ; }
do_data ()
{ command list ; }
cleanup ()
{ command list ; }
errors ()
{ command list ; }
setup
do_data
cleanup
# end program
技巧
. 在對(duì)函數(shù)命名時(shí)最好能使用有含義的名字即函數(shù)名能夠比較準(zhǔn)
確的描述函數(shù)所
完成
的任務(wù)
. 為了程序的維護(hù)方便請(qǐng)盡可能使用注釋
使用函數(shù)的另一個(gè)好處就是可以在一個(gè)程序中的不同地方執(zhí)行相同的
命令序列(函數(shù)),
如下所示
iscontinue()
{
while true
do
echo -n "Continue?(Y/N)"
read ANSWER
case $ANSWER in
[Yy]) return 0;;
[Nn]) return 1;;
*) echo "Answer Y or N";;
esac
done
}
這樣可以在shell編程中調(diào)用iscontinue確定是否繼續(xù)執(zhí)行
if iscontinue
then
continue
else
break
fi
** shell函數(shù)與shell程序非常相似但二者有一個(gè)非常重要的差別shell程序是由子sh
ell
執(zhí)行的而shell函數(shù)則是作為當(dāng)前shell的一部分被執(zhí)行的因此在
當(dāng)前shell中可以
改
變函數(shù)的定義此外在任意shell(包括交互式的shell)中均可定義函數(shù)
如
$ dir
dir: not found
$ dir () { ls -l ;}
$ dir
total 5875
-rw-r--r-- 1 hbwork 100 Nov 10 23:16 doc
-rw-r--r-- 1 hbwork 2973806 Nov 10 23:47 ns40docs.zip
-rw-r--r-- 1 hbwork 1715011 Nov 10 23:30 ns840usr.pdf
-rw-r--r-- 1 hbwork 1273409 Sep 23 1998 radsol21b6.tar.
Z
-rw-r--r-- 1 hbwork 7526 Nov 10 23:47 wget-log
-rw-r--r-- 1 hbwork 1748 Nov 13 21:51 wget-log.1
$ unset dir
$ dir () {
> echo "Permission Link Owner Group File_SZ LastAccess
FileName"
> echo "-----------------------------------------------------------"
> ls -l $*;
> }
$ dir
Permission Link Owner Group File_SZ LastAccess FileName
-----------------------------------------------------------
total 5875
-rw-r--r-- 1 hbwork 100 Nov 10 23:16 doc
-rw-r--r-- 1 hbwork 2973806 Nov 10 23:47 ns40docs.zip
-rw-r--r-- 1 hbwork 1715011 Nov 10 23:30 ns840usr.pdf
-rw-r--r-- 1 hbwork 1273409 Sep 23 1998 radsol21b6.tar.Z
-rw-r--r-- 1 hbwork 7526 Nov 10 23:47 wget-log
-rw-r--r-- 1 hbwork 1748 Nov 13 21:51 wget-log.1
通常情況下shell script是在子shell中執(zhí)行的困此在此子shell中
對(duì)變量所作的
修改對(duì)父shell不起作用點(diǎn)(.) 命令使用shell在不創(chuàng)建子shell而由
當(dāng)前shell讀取
并執(zhí)行一個(gè)shell script, 可以通過(guò)這種方式來(lái)定義函數(shù)及變量此外
點(diǎn)(.)命令最
常用的功能就是通過(guò)讀取.profile來(lái)重新配置初始化login變量如下
所示
$ . .profile
(csh相對(duì)于.命令的是source命令).
(7)使用And/Or結(jié)構(gòu)進(jìn)行有條件的命令執(zhí)行
<1> And , 僅當(dāng)?shù)谝粋€(gè)命令成功時(shí)才有執(zhí)行后一個(gè)命令,如同在邏輯與表
達(dá)式中如果前面的
結(jié)果為真時(shí)才有必要繼續(xù)運(yùn)算否則結(jié)果肯定為假
格式如下
command1 && command2
例rm $TEMPDIR/* && echo "File successfully removed"
等價(jià)于
if rm $TEMPDIR/*
then
echo "File successfully removed"
fi
<2>Or, 與AND相反僅當(dāng)前一個(gè)命令執(zhí)行出錯(cuò)時(shí)才執(zhí)行后一條命令
例 rm $TEMPDIR/* || echo "File not removed"
等價(jià)與
if rm $TEMPDIR/*
then
command
else
echo "File not removed"
fi
<3> 混合命令條件執(zhí)行
command1 && command2 && command3
當(dāng)command1, command2成功時(shí)才執(zhí)行command3
command1 && command2 || comamnd3
僅當(dāng)command1成功command2失敗時(shí)才執(zhí)行command3
當(dāng)然可以根據(jù)自己的需要進(jìn)行多種條件命令的組合在此不多講述
(8) 使用getopts命令讀取unix格式選項(xiàng)
UNIX格式選項(xiàng)指如下格式的命令行參數(shù)
command -options parameters
使用格式
getopts option_string variable
具體使用方法請(qǐng)參考getopts的在線文檔(man getopts).
示例如下
#newdate
if [ $# -lt 1 ]
then
date
else
while getopts mdyDHMSTjJwahr OPTION
do
case $OPTION
in
m) date '+%m ';; # Month of Year
d) date '+%d ';; # Day of Month
y) date '+%y ';; # Year
D) date '+%D ';; # MM/DD/YY
H) date '+%H ';; # Hour
M) date '+%M ';; # Minute
S) date '+%S ';; # Second
T) date '+%T ';; # HH:MM:SS
j) date '+%j ';; # day of year
J) date '+%y%j ';;# 5 digit Julian date
w) date '+%w ';; # Day of the Week
a) date '+%a ';; # Day abbreviation
h) date '+%h ';; # Month abbreviation
r) date '+%r ';; # AM-PM time
\?) echo "Invalid option $OPTION";;
esac
done
fi
$ newdate -J
94031
$ newdate -a -h -d
Mon
Jan
31
$ newdate -ahd
Mon
Jan
31
$
示例程序復(fù)制程序
# Syntax: duplicate [-c integer] [-v] filename
# where integer is the number of duplicate copies
# and -v is the verbose option
COPIES=1
VERBOSE=N
while getopts vc: OPTION
do
case $OPTION
in
c) COPIES=$OPTARG;;
v) VERBOSE=Y;;
\?) echo "Illegal Option"
exit 1;;
esac
done
if [ $OPTIND -gt $# ]
then
echo "No file name specified"
exit 2
fi
shift `expr $OPTIND -1`
FILE=$1
COPY=0
while [ $COPIES -gt $COPY ]
do
COPY=`expr $COPY + 1`
cp $FILE ${FILE}${COPY}
if [ VERBOSE = Y ]
then
echo ${FILE}${COPY}
fi
done
$ duplicate -v fileA
fileA1
$ duplicate -c 3 -v fileB
fileB1
fileB2
fileB3