shell的方法在相同的进程内执行,与调用它的脚本一致。对于方法来说,脚本中的所有变量均可见,且不需要执行export。方法中可以创建局部变量,且不影响正在调用的脚本。
1. 定义语法
(1) KornShell中的定义格式为:function name <复合命令>
(2) Bourne shell中的定义格式为: name() <复合命令>
(3) bash允许的格式:function name() <复合命令>
一个参数设置方法的返回码,若没有参数,方法的退出码默认为执行的最后一个命令执行结果。
local为shell中的内置命令,可用于限制方法(以及其子方法)中变量的作用域,但父进程中的变量不会改变。当参数扩展时,使用IFS而非空格将会导致分词。Bash4.0中, local和declare存在一个选项 -A,用于声明关联数组。
例:判断ip是否有效
## 检测Ip是否有效
isvalidip(){
case $ in
"" | *[!-.]* | *[!-]) return ;; ## 空值、非法字符、不以数字结尾,均不符合Ip
esac ## 将IFS设置为点号,但仅限于该方法中
local IFS=. ## 将Ip设为位置参数,分词之后,每个元素变成了参数
set -- $ ## 必须有4个参数,每个元素必须小于256,参数为空时,默认为666
[ $# -eq ] &&
[ ${:-} -le ] &&
[ ${:-} -le ] &&
[ ${:-} -le ] &&
[ ${:-} -le ] &&
}
使用source命令使脚本中的方法在当前shell中有效:. isvalidip
$ for ip in 127.0.0.1 168.260.0.234 123.100.34.32 204.255.122.150
> do
> if isvalidip "$ip"
> then
> printf "%15s: valid\n" "$ip"
> else
> printf "%15s: invalid\n" "$ip"
> fi
> done 127.0.0.1: valid
168.260.0.234: invalid
123.100.34.32: valid
204.255.122.150: valid
2. 复合命令
复合命令可以是一组封装在( ... )或{ ... }中的命令,封装在(( ... ))或[[ ... ]]中的表达式,或者shell的关键词块(case, for, while, select, until)。
例:检查有效整型
valint() #@ USAGE: valint INTEGER
case ${#-} in ## 接收负数
*[!-]*) echo false;; ## 包含非数字字符
*) echo true;;
esac
若方法的主体由引号包含,则它将在子shell中执行,并且执行期间产生的变化在退出时不再有效。
$ funky() (nam=nobody; echo "name = $name")
name=Rempelstilskin
$ funky
name = nobody
$ echo "name = $name"
name = Rempelstilskin
3. 获取结果
1) 设置不同的退出码
例:检验整型是否在特定范围
rangecheck() #@ USAGE: rangecheck int [low [high]]
if [ "$1" -lt ${:-} ];then ## 数值太小返回1,若无第二个参数,默认为10
return
elif [ "$1" -gt ${:-} ];then ## 数据太大返回0,若无第三个参数,默认为20
return
else
return
fi
2) 打印结果
例:打印环境变量信息
uinfo() #@ USAGE: uinfo [file]
{
printf "%12s: %s\n" \
USER "${USER:-No value assigned }" \
PWD "${PWD:-No value assigned}" \
COLUMNS "${COLUMNS:-No value assigned}" \
LINES "${LINES:-No value assigned}" \
SHELL "${SHELL:-No value assigned}" \
HOME "${HOME:-No value assigned}" \
TERM "${TERM:-No value assigned}"
} > ${:-/dev/fd/}
3) 结果保存在多个变量中
例:3个整数排序
_max3() #@ 对3个整数排序,并且分别保存在$_MAX3,$_MID3及$_MIN3中
{
[ $# -ne ] && return
[ $ -gt $ ] && { set -- $ $ $; }
[ $ -gt $ ] && { set -- $ $ $; }
[ $ -gt $ ] && { set -- $ $ $; }
_MAX3=$
_MID3=$
_MIN3=$
printf "%d\t%d\t%d\n" $_MAX3 $_MID3 $_MIN3
} max3() #@ 对3个整数排序并保存在array中
{
declare -n _max3=${:-_MAX3} #@ 当命令行未提供变量名,则默认使用_MAX3
(( $# < )) && return
(( $ > $ )) && set -- "$2" "$1" "$3"
(( $ > $ )) && set -- "$1" "$3" "$2"
(( $ > $ )) && set -- "$2" "$1" "$3"
_max3=( "$3" "$2" "$1" )
}
4. 示例脚本
##
## 设置默认
##
prompt=" ==> "
template='<!DOCTYPY html>
<html lang="en">
<head>
<meta charset=utf->
<title>%s</title>
<link href=%s" rel="stylesheet">
</head>
<body>
<h1>%s</h1>
<div id=main></div>
</body>
</html>
' ##
## 定义shell函数
##
die(){ #@ 描述:打印错误信息,并且以ERROR退出
error=$
shift
[ -n "$*" ] && printf "%s\n" "$*" >&
exit "$error"
} usage(){ #@ 打印脚本使用用途
printf "USAGE: %s HTMLFILE\n" "$progname"
} version(){
printf "%s version %s: " "$progname" "${version:-1}"
} bashversion=${BASH_VERSION%%.*}
if [ ${bashversion:-} -ge ];then
## bash4.x 的read有-i选项,用于提供一个初始值(如下的$)
readline(){
read -ep "${2:-"$prompt"}" -i "$3" "$1"
}
elif [ ${BASHVERSION:-} -ge ];then
readline(){
history -s "$3"
printf "Press up arror to edit default value: '%s'\n" "${2:-none}"
read -ep "${2:-"$prompt"}" "$1"
}
else
readline(){
printf "Press enter for default of '%s'\n" "$3"
printf "%s " "${2:-"$prompt"}"
read
eval "$1=\${REPLY:-"$"}"
}
fi if [ $# -ne ];then
usage
exit
fi filename=$ readline title "Page title: "
readline h1 "Main headline: " "$title"
readline css "Style sheet file: " "${filename%.*}.css" printf "$template" "$title" "$css" "$h1" > "$filename"
执行示例结果:
$ bash test.sh test
Page title: hello
Main headline: hello
Style sheet file: test.css
$ cat test
<!DOCTYPY html>
<html lang="en">
<head>
<meta charset=utf->
<title>hello</title>
<link href=test.css" rel="stylesheet">
</head>
<body>
<h1>hello</h1>
<div id=main></div>
</body>
</html>