如何在shell中执行一个模块,这个模块需要shell 循环传入参数数

73823人阅读
shell 脚本(10)
&参数处理-Shell传入参数的处理1. $# 传递到脚本的参数个数2. $* 以一个单字符串显示所有向脚本传递的参数。与位置变量不同,此选项参数可超过9个3. $$ 脚本运行的当前进程ID号4. $! 后台运行的最后一个进程的进程ID号5. $@ 与$#相同,但是使用时加引号,并在引号中返回每个参数6. $- 显示shell使用的当前选项,与set命令功能相同7. $? 显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误。变量 含义 $0 脚本名字 $1 位置参数 #1 $2 - $9 位置参数 #2 - #9 ${10} 位置参数 #10 $# 位置参数的个数 "$*" 所有的位置参数(作为单个字符串) * "$@" 所有的位置参数(每个都作为独立的字符串) ${#*} 传递到脚本中的命令行参数的个数 ${#@} 传递到脚本中的命令行参数的个数 $? 返回值 $$ 脚本的进程ID(PID) $- 传递到脚本中的标志(使用set) $_ 之前命令的最后一个参数 $! 运行在后台的最后一个作业的进程ID(PID)
&&&&&& 使用shell处理的时候对参数的处理是个基本模块,所以今天找到一篇简单易懂的文章来进行参考,作为以后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/bash2 3 for arg in "$*"4 do5&&&& echo $arg6 done7 8 for arg in "$@"9 do10&&&& echo $arg11 done12
执行./test.sh -f config.conf -n 10 会打印:
-f config.conf -n 10&&& #这是"$*"的输出
-f&& #以下为$@的输出
config.conf
&& 所以,手工处理的方式即对这些变量的处理。因为手工处理高度依赖于你在命令行上所传参数的位置,所以一般都只用来处理较简单的参数。如
&& ./test.sh 10
&& 而很少使用./test -n 10这种带选项的方式。 典型用法为:
#!/bin/bash
if [ x$1 != x ]then&&& #...有参数elsethen&&& #...没有参数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非常简单:代码
#!/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&&&&&&& ;;&&&&&&& esacdone
现在就可以使用:./test.sh -a arg -b -c 或./test.sh -a arg -bc来加载了。应该说绝大多数脚本使用该函数就可以了,如果需要支持长选项以及可选参数,那么就需要使用getopt.下面是getopt自带的一个例子:
#!/bin/bash
# A small example program for using the new getopt(1) program.# This program will only work with bash(1)# An similar program using the tcsh(1) script 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的处理,下面处理具体选项。
do&&&&&&& case "$1" in&&&&&&&&&&&&&&& -a|--a-long) echo "Option a" ;;&&&&&&&&&&&&&&& -b|--b-long) echo "Option b, argument /`$2'" ; shift 2 ;;&&&&&&&&&&&&&&& -c|--c-long)&&&&&&&&&&&&&&&&&&&&&&& # c 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 ;;&&&&&&&&&&&&&&&&&&&&&&&;&&&&&&&&&&&&&&& --);&&&&&&&&&&&&&&& *) echo "Internal error!" ; exit 1 ;;&&&&&&& esacdoneecho "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较复杂,功能也更强大。
******************************************************************************************************************************
shift其实很简单的,就是左移参数列表,shift一次就将最左边的参数$1移出去了,然后 原来的$2现在就变成了$1。
shift后面还可以带上一个数字,指明要移出多少个参数(默认只移出一个),比如说 shift 3 就是移出3个参数,之后原来的$4就变成了现在的$1。
eval就是先将后面的参数执行一遍,将必要的置换都做了,再来执行命令。举个例子: MYFILE="cat myfile" echo $MYFILE&& # output: cat myfile eval $MYFILE&& # output: contents of myfile
再举个详细点儿的例子: #!/bin/bash # evalit echo " Total number of arguments passed: $#" echo " The process ID: $$" echo " Last argument: " $(eval echo /$$#) 运行脚本: $ ./evalit alpha bravo charlie output as follows: Total number of arguments passed: 3 The process ID: 780 Last argument:&& charlie
读取文件for i in `cat abc.txt`doecho $idone
参考知识库
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
访问:148742次
积分:1313
积分:1313
排名:千里之外
原创:35篇
转载:19篇
(1)(1)(1)(1)(3)(3)(8)(3)(3)(4)(3)(3)(4)(10)(6)新手园地& & & 硬件问题Linux系统管理Linux网络问题Linux环境编程Linux桌面系统国产LinuxBSD& & & BSD文档中心AIX& & & 新手入门& & & AIX文档中心& & & 资源下载& & & Power高级应用& & & IBM存储AS400Solaris& & & Solaris文档中心HP-UX& & & HP文档中心SCO UNIX& & & SCO文档中心互操作专区IRIXTru64 UNIXMac OS X门户网站运维集群和高可用服务器应用监控和防护虚拟化技术架构设计行业应用和管理服务器及硬件技术& & & 服务器资源下载云计算& & & 云计算文档中心& & & 云计算业界& & & 云计算资源下载存储备份& & & 存储文档中心& & & 存储业界& & & 存储资源下载& & & Symantec技术交流区安全技术网络技术& & & 网络技术文档中心C/C++& & & GUI编程& & & Functional编程内核源码& & & 内核问题移动开发& & & 移动开发技术资料ShellPerlJava& & & Java文档中心PHP& & & php文档中心Python& & & Python文档中心RubyCPU与编译器嵌入式开发驱动开发Web开发VoIP开发技术MySQL& & & MySQL文档中心SybaseOraclePostgreSQLDB2Informix数据仓库与数据挖掘NoSQL技术IT业界新闻与评论IT职业生涯& & & 猎头招聘IT图书与评论& & & CU技术图书大系& & & Linux书友会二手交易下载共享Linux文档专区IT培训与认证& & & 培训交流& & & 认证培训清茶斋投资理财运动地带快乐数码摄影& & & 摄影器材& & & 摄影比赛专区IT爱车族旅游天下站务交流版主会议室博客SNS站务交流区CU活动专区& & & Power活动专区& & & 拍卖交流区频道交流区
UID9860515空间积分0 积分159阅读权限10帖子精华可用积分159 信誉积分0 专家积分0 在线时间166 小时注册时间最后登录
白手起家, 积分 159, 距离下一级还需 41 积分
帖子主题精华可用积分159 信誉积分0 专家积分0 在线时间166 小时注册时间最后登录
论坛徽章:0
请问我执行一个shell脚本,内容顺序执行“/data/bin/module1& & /data/bin/module2 ”,即执行完模块1,再执行模块2,请问在脚本执行时,如何获取shell脚本的pid以及里头模块的pid?
&&nbsp|&&nbsp&&nbsp|&&nbsp&&nbsp|&&nbsp&&nbsp|&&nbsp
UID空间积分802 积分1392阅读权限30帖子精华可用积分1392 信誉积分204 专家积分0 在线时间4309 小时注册时间最后登录
家境小康, 积分 1392, 距离下一级还需 608 积分
帖子主题精华可用积分1392 信誉积分204 专家积分0 在线时间4309 小时注册时间最后登录
论坛徽章:0
UID8019015空间积分809 积分2006阅读权限50帖子精华可用积分2007 信誉积分140 专家积分70 在线时间1501 小时注册时间最后登录
小富即安, 积分 2006, 距离下一级还需 2994 积分
帖子主题精华可用积分2007 信誉积分140 专家积分70 在线时间1501 小时注册时间最后登录
论坛徽章:0
long time no see ~
UID1082空间积分0 积分2398阅读权限50帖子精华可用积分2398 信誉积分940 专家积分112 在线时间6792 小时注册时间最后登录
小富即安, 积分 2398, 距离下一级还需 2602 积分
帖子主题精华可用积分2398 信誉积分940 专家积分112 在线时间6792 小时注册时间最后登录
论坛徽章:0
ps -ef|grep 模块名
2.6.23.1-42.fc8 #1 SMP Tue Oct 30 13:55:12 EDT
i686 i386 GNU/Linux
SunOS s_test 5.8 Generic_ sun4u sparc SUNW,Ultra-60
UID1082空间积分0 积分2398阅读权限50帖子精华可用积分2398 信誉积分940 专家积分112 在线时间6792 小时注册时间最后登录
小富即安, 积分 2398, 距离下一级还需 2602 积分
帖子主题精华可用积分2398 信誉积分940 专家积分112 在线时间6792 小时注册时间最后登录
论坛徽章:0
原帖由 我是DBA 于
11:29 发表
那个貌似是父进程的的吧?
2.6.23.1-42.fc8 #1 SMP Tue Oct 30 13:55:12 EDT
i686 i386 GNU/Linux
SunOS s_test 5.8 Generic_ sun4u sparc SUNW,Ultra-60
UID8019015空间积分809 积分2006阅读权限50帖子精华可用积分2007 信誉积分140 专家积分70 在线时间1501 小时注册时间最后登录
小富即安, 积分 2006, 距离下一级还需 2994 积分
帖子主题精华可用积分2007 信誉积分140 专家积分70 在线时间1501 小时注册时间最后登录
论坛徽章:0
回复 #5 welcome008 的帖子
对哦,应该用你的方法
long time no see ~
UID9860515空间积分0 积分159阅读权限10帖子精华可用积分159 信誉积分0 专家积分0 在线时间166 小时注册时间最后登录
白手起家, 积分 159, 距离下一级还需 41 积分
帖子主题精华可用积分159 信誉积分0 专家积分0 在线时间166 小时注册时间最后登录
论坛徽章:0
原帖由 welcome008 于
11:29 发表
ps -ef|grep 模块名
这只是列出含有模块的进程,不能获得模块pid啊?
而$$是获取shell版本的pid...继续等待中。。。
网上找的是:
ps -ef |grep cmos |awk '{print $2}' |while read pid
& && &&&do
& && && && && & kill -9 $pid
& && &&&done
可是我的却提示awk找不到。。。。
UID8019015空间积分809 积分2006阅读权限50帖子精华可用积分2007 信誉积分140 专家积分70 在线时间1501 小时注册时间最后登录
小富即安, 积分 2006, 距离下一级还需 2994 积分
帖子主题精华可用积分2007 信誉积分140 专家积分70 在线时间1501 小时注册时间最后登录
论坛徽章:0
回复 #7 wolfme 的帖子
就是这样,你系统没有awk?
long time no see ~
UID9860515空间积分0 积分159阅读权限10帖子精华可用积分159 信誉积分0 专家积分0 在线时间166 小时注册时间最后登录
白手起家, 积分 159, 距离下一级还需 41 积分
帖子主题精华可用积分159 信誉积分0 专家积分0 在线时间166 小时注册时间最后登录
论坛徽章:0
回复 #8 我是DBA 的帖子
没有,我是在linux系统手持设备中运行,可以运行pidof 模块名来回去模块pid,如何将pid保存到变量中呢?这样写为什么不行?
#!/bin/bash
pidof cmos &&pid
kill&&$pid
UID8019015空间积分809 积分2006阅读权限50帖子精华可用积分2007 信誉积分140 专家积分70 在线时间1501 小时注册时间最后登录
小富即安, 积分 2006, 距离下一级还需 2994 积分
帖子主题精华可用积分2007 信誉积分140 专家积分70 在线时间1501 小时注册时间最后登录
论坛徽章:0
pid=`pidof cmos`
long time no see ~
北京皓辰网域网络信息技术有限公司. 版权所有 京ICP证:060528号 北京市公安局海淀分局网监中心备案编号:
广播电视节目制作经营许可证(京) 字第1234号
中国互联网协会会员&&联系我们:
感谢所有关心和支持过ChinaUnix的朋友们
转载本站内容请注明原作者名及出处ansible(14)
ansible命令执行模块使用
1、命令执行模块-command
在远程节点上运行命令。
命令模块使用命令名称,接上空格-的分割符作为参数使用,但是不支持管道符和变量等,如果要使用这些,那么可以使用shell模块。
当文件名已经存在的时候,此步不会执行
Executable
换用shell执行命令,表示为一个绝对路径
需要执行的脚本(没有真正的参数为free_form)
当文件名称存在的时候,此步不会执行
如果在ansible.cfg中存在告警,如果设定了no/False,不会警告此行
1.1 列出目录的文件
[root@ansibleserver tmp]# ansible pythonserver -a &ls -l chdir=/tmp creates=kel.conf&
SSH password:
192.168.1.60 | success | rc=0 &&
-rw------- 1 root root 29 Jan 19 15:13 kel
-rw-r--r-- 1 root root 16 Jan 19 15:13 kel.@15:13~
在这个里面,首先更换目录到tmp目录中,然后查看kel.conf是否存在,如果存在,那么命令不会执行;如果不存在,那么执行命令,在这里也可以看到,命令是必须存在的,但是没有参数名为free_form参数
1.2 切换目录创建文件
[root@ansibleserver tmp]# ansible pythonserver -a &touch kel.conf chdir=/tmp&
SSH password:
192.168.1.60 | success | rc=0 &&
以上的命令表示为更换目录到tmp中,然后创建文件kel.conf
在使用ansible中的时候,默认的模块是-m command,从而模块的参数不需要填写,直接使用即可。
1.3 不支持管道符等操作
[root@ansibleserver tmp]# ansible pythonserver -a &cat /tmp/kel.conf &/tmp/111&
SSH password:
192.168.1.60 | FAILED | rc=1 &&
cat: &/tmp/111: No such file or directory
目录和文件都是存在的,但是依然报错不存在目录或者文件,从而在command命令中是不支持管道符和其他的一些操作的,在使用这些命令的时候,可以选择使用shell模块
2、 shell模块
在远程节点上执行命令。在执行命令的时候使用的是/bin/sh。
修改当前目录
当文件名已经存在的时候,此步不会执行
Executable
换用shell执行命令,表示为一个绝对路径
需要执行的脚本(没有真正的参数为free_form)
当文件名称存在的时候,此步不会执行
如果在ansible.cfg中存在告警,如果设定了no/False,不会警告此行
2.1 执行命令保存输出
[root@ansibleserver tmp]# ansible pythonserver -m shell -a &sh /tmp/kel.sh &&/tmp/kel.log&
SSH password:
192.168.1.60 | success | rc=0 &&
执行远程机器上的脚本,脚本目录为/tmp/kel.sh,然后将执行命令的结果存放在路径/tmp/kel.log中,注意在进行保存文件的时候,写上全路径,否则就会保存在登录之后的默认路径中。
2.2 换成路径然后执行命令
[root@ansibleserver tmp]# ansible pythonserver -m shell -a &sh /tmp/kel.sh &&kelly.log chdir=/tmp&
SSH password:
192.168.1.60 | success | rc=0 &&
可以看到修改之后的目录,然后再执行命令。
3、脚本执行模块-script
将script传送到远程主机之后,执行。此模块的执行,在远程主机上,不需要python环境
当文件名已经存在的时候,此步不会执行
需要执行的脚本(没有真正的参数为free_form)
当文件名称存在的时候,此步不会执行
3.1 在远程主机上执行脚本
[root@ansibleserver tmp]# ansible pythonserver -m script -a &/tmp/kel.sh &/tmp/kelkel.log&
SSH password:
192.168.1.60 | success && {
&&& &changed&: true,
&&& &rc&: 0,
&&& &stderr&: &&,
&&& &stdout&: &&
脚本kel.sh在本机上,然后将脚本远程传送到pythonserver主机中,然后执行这个脚本,从而将执行的结果保存在全路径的文件kelkel.log中,在保存文件的时候,最好用全路径
4、 ssh命令执行模块-raw
此模块的执行,在远程主机上,不需要python环境,主要是用来执行脏的ssh命令。
需要执行的脚本(没有真正的参数为free_form)
当文件名称存在的时候,此步不会执行
主要使用raw的原因是,如果老版本的python中,那么就需要用raw;如果客户端是路由器的话,那么没有python模块的话,那么就必须要使用raw模块
raw很多地方和shell和command相似,更多推荐的地方使用的是shell和command模块
参考知识库
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
访问:105238次
积分:3010
积分:3010
排名:第7856名
原创:202篇
(1)(20)(25)(7)(12)(16)(1)(9)(11)(10)(3)(2)(1)(10)(18)(3)(1)(4)(8)(2)(10)(5)(1)(4)(4)(8)(1)(3)(4)(1)(1)(1)【图文】第2章 构造和运行模块_百度文库
两大类热门资源免费畅读
续费一年阅读会员,立省24元!
评价文档:
第2章 构造和运行模块
上传于||暂无简介
大小:521.50KB
登录百度文库,专享文档复制特权,财富值每天免费拿!
你可能喜欢实验三 模块编程_百度文库
两大类热门资源免费畅读
续费一年阅读会员,立省24元!
实验三 模块编程
上传于||文档简介
&&g​o​o​d
阅读已结束,如果下载本文需要使用0下载券
想免费下载更多文档?
下载文档到电脑,查找使用更方便
还剩18页未读,继续阅读
你可能喜欢

我要回帖

更多关于 shell 如何传入参数 的文章

 

随机推荐