getopts详解

getopts详解

(2013-08-29 12:41:09)
转载
标签:

shell

getopts

选项

参数

分类: 脚本编程

写程序的时候经常要处理命令行参数,本文描述在Bash下的命令行处理方式。

   选项与参数:

   如下一个命令行:

./test.sh -f config.conf -v --prefix=/home


   我们称-f为选项,它需要一个参数,即config.conf, -v也是一个选项,但它不需要参数。

  --prefix我们称之为一个长选项,即选项本身多于一个字符,它也需要一个参数,用等号连接,当然等号不是必须的,/home可以直接写在--prefix后面,即--prefix/home,更多的限制后面具体会讲到。
   在bash中,可以用以下三种方式来处理命令行参数,每种方式都有自己的应用场景。

   * 手工处理方式
    *getopts
    *getopt

   下面我们依次讨论这三种处理方式。

1. 手工处理方式


   在手工处理方式中,首先要知道几个变量,还是以上面的命令行为例:

     $0 :./test.sh,即命令本身,相当于C/C++中的argv[0]
     $1 :-f,第一个参数.
     $2 :config.conf
      $3, $4... :类推。
     $#  参数的个数,不包括命令本身,上例中$#为4.
     $@ :参数本身的列表,也不包括命令本身,如上例为-f config.conf -v --prefix=/home
     $* :和$@相同,但"$*" 和 "$@"(加引号)并不同,"$*"将所有的参数解释成一个字符串,而"$@"是一个参数数组。如下例所示:

复制代码

 1 #!/bin/bash
 2 
 3 for arg in "$*"
 4 do
 5     echo $arg
 6 done
 7 
 8 for arg in "$@"
 9 do
10     echo $arg
11 done
12 

复制代码

 


执行./test.sh -f config.conf -n10 会打印:

-f config.conf -n10   #这是"$*"的输出

-f   #以下为$@的输出

config.conf

-n

10

 

   所以,手工处理的方式即对这些变量的处理。因为手工处理高度依赖于你在命令行上所传参数的位置,所以一般都只用来处理较简单的参数。如

   ./test.sh 10

   而很少使用./test -n 10这种带选项的方式。典型用法为:

复制代码

#!/bin/bash

if x$1 != ]
then
    #...
有参数
else
then
    #...
没有参数
fi

复制代码



为什么要使用x$1 !=x 这种方式来比较呢?想像一下这种方式比较:


if [ -n $1 ]  #$1不为空

但如果用户不传参数的时候,$1为空,这时就会变成[ -n ] ,所以需要加一个辅助字符串来进行比较。

手工处理方式能满足大多数的简单需求,配合shift使用也能构造出强大的功能,但在要处理复杂选项的时候建议用下面的两种方法。
 

2. getopts/getopt


处理命令行参数是一个相似而又复杂的事情,为此,C提供了getopt/getopt_long等函数,
C++的boost提供了Options库,在shell中,处理此事的是getopts和getopt.

getopts和getopt功能相似但又不完全相同,其中getopt是独立的可执行文件,而getopts是由Bash内置的。

先来看看参数传递的典型用法:

    * ./test.sh-a -b -c  : 短选项,各选项不需参数
    * ./test.sh-abc   : 短选项,和上一种方法的效果一样,只是将所有的选项写在一起。
    * ./test.sh-a args -b -c :短选项,其中-a需要参数,而-b -c不需参数。
    * ./test.sh--a-long=args --b-long :长选项

我们先来看getopts,它不支持长选项。

使用getopts非常简单:
代码

复制代码

#test.sh

#!/bin/bash

while getopts "a:bc" arg #选项后面的冒号表示该选项需要参数
do
        
case $arg in
             a)
                echo 
"a's arg:$OPTARG" #参数存在$OPTARG
                ;;
             b)
                echo 
"b"
                ;;
             c)
                echo 
"c"
                ;;
             ?)  #
当有不认识的选项的时候arg?
            echo 
"unkonw argument"
        exit 
1
        ;;
        esac
done

复制代码



现在就可以使用:
./test.sh -a arg -b -c

./test.sh -a arg -bc
来加载了。
应该说绝大多数脚本使用该函数就可以了,如果需要支持长选项以及可选参数,那么就需要使用getopt.
下面是getopt自带的一个例子:

复制代码


#!/bin/bash

small example program 
for using the new getopt(1program.
This program will only work with bash(
1)
An similar program 
using the tcsh(1script language can be found
as parse.tcsh

Example input and output (from the bash prompt):
./parse.bash -a par1 
'another arg' --c-long 'wow!*\?' -cmore -b very long "
Option a
Option c, no argument
Option c, argument `more
'
Option b, argument very long '
Remaining arguments:
--> `par1
'
--> `another arg'
--> `wow!*\?'

Note that we use `
"$@"' to let each command-line parameter expand to a
separate word. The quotes around `$@are essential!
We need TEMP as the `eval set --would nuke the return value of getopt.

#-o
表示短选项,两个冒号表示该选项有一个可选参数,可选参数必须紧贴选项
#
-carg 而不能是-c arg
#--long
表示长选项
#
"$@"在上面解释过
-n:
出错时的信息
-- 
:举一个例子比较好理解:
#
我们要创建一个名字为 "-f"的目录你会怎么办?
mkdir -f #
不成功,因为-f会被mkdir当作选项来解析,这时就可以使用
mkdir -- -f 
这样-f就不会被作为选项。

TEMP=`getopt -o ab:c:: --
long a-long,b-long:,c-long:: \
     -n 
'example.bash' -- "$@"`

if $? != 0 then echo "Terminating..." >&2 exit 1 fi

Note the quotes around `$TEMP
': they are essential!
#set 会重新排列参数的顺序,也就是改变$1,$2...$n的值,这些值在getopt中重新排列过了
eval 
set -- "$TEMP"

#
经过getopt的处理,下面处理具体选项。

while true do
        
case "$1" in
                -a|--a-
longecho "Option a" shift ;;
                -b|--b-
longecho "Option b, argument \`$2'" shift 2 ;;
                -c|--c-
long)
                        has an optional argument. As we are 
in quoted mode,
                        an empty parameter will be generated 
if its optional
                        argument 
is not found.
                        
case "$2" in
                                
""echo "Option c, no argument"shift 2 ;;
                                *)  echo 
"Option c, argument \`$2'" shift 2 ;;
                        esac ;;
                --) shift 
break ;;
                *) echo 
"Internal error!" exit 1 ;;
        esac
done
echo 
"Remaining arguments:"
for arg do
   echo 
'--> '"\`$arg'" ;
done

复制代码



比如我们使用
./test -a  -b arg arg1 -c
你可以看到,命令行中多了个arg1参数,在经过getopt和set之后,命令行会变为:
-a -b arg -c -- arg1
$1指向-a,$2指向-b,$3指向arg,$4指向-c,$5指向--,而多出的arg1则被放到了最后。

3.总结

一般小脚本手工处理也许就够了,getopts能处理绝大多数的情况,getopt较复杂,功能也更强大。
有问题请指出,不胜感激。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
对于计算机专业的学生而言,参加各类比赛能够带来多方面的益处,具体包括但不限于以下几点: 技能提升: 参与比赛促使学生深入学习和掌握计算机领域的专业知识与技能,如编程语言、算法设计、软件工程、网络安全等。 比赛通常涉及实际问题的解决,有助于将理论知识应用于实践中,增强问题解决能力。 实践经验: 大多数比赛都要求参赛者设计并实现解决方案,这提供了宝贵的动手操作机会,有助于积累项目经验。 实践经验对于计算机专业的学生尤为重要,因为雇主往往更青睐有实际项目背景的候选人。 团队合作: 许多比赛鼓励团队协作,这有助于培养学生的团队精神、沟通技巧和领导能力。 团队合作还能促进学生之间的知识共享和思维碰撞,有助于形成更全面的解决方案。 职业发展: 获奖经历可以显著增强简历的吸引力,为求职或继续深造提供有力支持。 某些比赛可能直接与企业合作,提供实习、工作机会或奖学金,为学生的职业生涯打开更多门路。 网络拓展: 比赛是结识同行业人才的好机会,可以帮助学生建立行业联系,这对于未来的职业发展非常重要。 奖金与荣誉: 许多比赛提供奖金或奖品,这不仅能给予学生经济上的奖励,还能增强其成就感和自信心。 荣誉证书或奖状可以证明学生的成就,对个人品牌建设有积极作用。 创新与研究: 参加比赛可以激发学生的创新思维,推动科研项目的开展,有时甚至能促成学术论文的发表。 个人成长: 在准备和参加比赛的过程中,学生将面临压力与挑战,这有助于培养良好的心理素质和抗压能力。 自我挑战和克服困难的经历对个人成长有着深远的影响。 综上所述,参加计算机领域的比赛对于学生来说是一个全面发展的平台,不仅可以提升专业技能,还能增强团队协作、沟通、解决问题的能力,并为未来的职业生涯奠定坚实的基础。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值