Python,Perl,Bash命令行参数 Part I
日常经常性的和Perl,Python,Bash打交道,但是又经常性的搞混他们之间,在命令行上的特殊性和index的区别,Python真的是人性化到家了,但是命令行上就很原始,但是很有分类感。还是会喜欢Perl一点(至少还可以动脑子,不至于生锈),感觉Perl是个杂合子,而python像胡彦斌,bash像韩红(最近看我是歌手第三季,嘻嘻)。Bash就是家常菜了,虽然Perl可以大部分代替Bash的功能,像grep sort map等,但是某些时候真心很麻烦,还是直接在命令行中bash好一些和命令感觉好。[如不苟同,忍气吞声就好]
下面将是我很多一部分的其中一小部分的开始,由于不断的装系统,更换系统过程中造成了很多总结的丢失,现在又要重新开始整理,很恼火,但是,回想旧爱也是甜蜜的。
基础篇
Perl
默认的命令行参数保存在@ARGV数组中
获取用 $ARGV[0],$ARGV[1],… 来表示第1个,第2个参数,…
$0表示当前脚本的文件名,和Bash一样
注意:就即使你是perl script.pl A B C,$0也会是script.pl,而不会是perl
Python
是需要sys模块
import sys
sys.argv是一个list,列表(用sys.argv.__class__查看)
获取用sys.argv[1],sys.argv[2],… 来表示第1个,第2个参数,…
sys.argv[0]表示文件名
Bash
是直接用$n,$@,$*表示所有的参数[有区别,see below]
获取用$1,$2,… 来表示第1个,第2个参数
$0始终表示文件名,即使在函数中
bash的每个函数也是这个规律的,$@表示所有
$0,$1,$2分别表示第1个,第2个,第3个传递给函数的参数
> cat test.sh
1 #!/bin/bash
2 function aa() {
3 echo $0, $1,$2
4 }
5 aa I love you
结果:test.sh,I,love
提升篇
很多时候,要共享脚本,要写清楚参数的顺序才可以,不然天知道该怎么传参数,如果看代码也不知道什么顺序。可以用注释注明顺序,或者写个usage()和简单的-h选项也可以,但谁能保证你的代码就可以被正常运行的,所以嘛,还是用getopts比较好,Linux/Unix有原生的getopt(s)命令的,Python和Perl也有相应的模块可以使用。
Bash
先看一个例子(getopts)
> cat vallis.sh
1 YON="don’t love"
2 while getopts 'F:nO:t:' OPT; do
3 case $OPT in
4 F) Who=$OPTARG;;
5 n) YON='love';;
6 O) TGT=$OPTARG;;
7 t) TL=$OPTARG;;
8 ?) echo '-h for help';;
9 esac
10 done
11 echo $Who $YON $TGT $TL
12 shift $(($OPTIND - 1))
13 echo $*
> sh untitled.sh -F I -O you -t baby -n forever
结果是
I love you baby
forever
解析一下就是
先是一个while框架循环
while …;do
…
done
再是 getopts 'F:nO:t:’ OPT 就是按optstring提取相应的值到$OPTARG中
每次只提取一个,所以使用到while循环,当然可以使用for loop了
optstring部分的'F:nO:t:’
- 每个字母代表一个短选项(short option),带冒号:的表示有选项值的,不带的表示没有值,相当于开关true/false
- 不在optstring中的会给出警告信息,可以在最前面加上:就可以了
- 2个很重要的参数是OPTARG和OPTINT(1-based),$OPTARG表示当前选项的值,$OPTINT表示在参数列表中的位移
- 选项(-)和选项值之间的空格是可有可无的(Mac OS X 10.10下成功)
- 不支持长选项(—)
- option部分必须写在其他参数之前,从OPTINT看出
紧接着是case框架分选
case $i in
…) statement1
statement2;; #注意每个块(block)结束的时候要用;;分割
…
esac
shift $(($OPTIND - 1))是将argv中的剩下的不是opt部分(opt部分已经处理过了)的参数重新set为$1,$2,…
$*,$@,这样剩下的参数可以用$n来访问,也就是说,删去了前面的option部分
上面的结果中的forever就是这样的
如果不用shift的话,$1仍然是-F,$2是I,...
总体上bash的这个和linux C语言的一样繁琐,只适合简单的选项设置
Another
还有一个getopt命令,是bash内置的(built-ins),上面的getopts是外置的(not built-ins)
区别在于getopt的参数是有-的(leading dash)并且和set结合使用
> cat valli.sh
1 YON="don’t love"
2 args=`getopt F:nO:t: $*`
3 for i; do
4 case $i in
5 -F)
6 shift # shift out -F then $1 is its value
7 Who=$1
8 shift;;
9 # can be replaced by
10 # Who=$2
11 # shift 2;;
12 # appliable to -O and -t or any option with a value
13 -n)
14 YON='love'
15 shift;;
16 -O)
17 shift
18 TGT=$1
19 shift;;
20 -t)
21 shift
22 TL=$1
23 shift;;
24 esac
25 done
26 echo $Who $YON $TGT $TL
27 echo $*
或者
> cat valli2.sh
1 YON="don’t love"
2 args=`getopt F:nO:t: $*`
3 set -- $args # but I think useless [5]
4 for i; do
5 case "$i" in
6 -F)
7 Who=$2
8 shift 2;;
9 -n)
10 YON='love'
11 shift;;
12 -O)
13 TGT=$2
14 shift 2;;
15 -t)
16 TL=$2
17 shift 2;;
18 --)
19 shift
20 break;;
21 esac
22 done
23 echo $Who $YON $TGT $TL
24 echo $*
同样成功
好原始,真的就和linux C语言没什么区别
注意
- getopt在Mac OS X 10.10反正不支持—long -n -o 等getopt参数[4],具体可以看reference:4
- getopt是内置的命令,getopts只能在shell script中
Perl
perl的getopt是常用的Getopt::Long,Getopt::Std模块,这个就比较智能了,排除不兼容[6]
先看一下例子
1 use Getopt::Std;
2
3 %options=();
4 getopts("F:O:nt:", \%options); #high spot
5 $does="don't love";
6 print "$options{F} " if defined $options{F};
7 if (defined $options{n}){
8 $does='love';
9 print "$does ";
10 }
11 print "$options{O} " if defined $options{O};
12 print "$options{t}\n" if defined $options{t};
13 print "@ARGV\n”;
就像写英语作文一样顺畅自如,仿佛在交流。但是着实是个话痨。
开始的%options哈希表的key为option,value为option value
Python
就是optaprse以及更新版argparse,但是更习惯用optparse,尽管2.7版之后不再开发。