Ruby和Linux Shell共同编程的示例
有了shell为毛还要ruby呢?话不能这么说,有些小功能用ruby还是很方便的,比如说字符串的反转再加1功能用shell来写就比较麻烦.str="123456",我们定义一个反转后再+1的小方法:
2.1.5 :020 > class String
2.1.5 :021?> def r_add_1
2.1.5 :022?> replace reverse.each_byte.map {|byte|byte+1}.pack("c*")
2.1.5 :023?> end
2.1.5 :024?> end
执行后str变为:
2.1.5 :026 > "123456".r_add_1
=> "765432"
下面我们写一个脚本把shell中的变量r_add_1一下:
#!/opt/local/bin/bash
export id="abc123456789" #just
cat <<EOF | ruby |read ret
class String
def r_add_1
replace reverse.each_byte.map {|b|b+1}.pack("c*")
end
end
puts ENV["id"].dup.r_add_1
EOF
执行一下看看:
linux@linuxidc: ruby_src$./read.sh
linux@linuxidc: ruby_src$echo $ret
linux@linuxidc: ruby_src$
咦,为毛ret变量为空值啊?道理很简单:管道2侧是2个不同进程,加上主bash的进程,一共是3个进程。你不用指望read ret进程建立的变量会在主bash进程中生效啊!一种解决办法是用小括号强制read ret和echo在一个进程中执行以获取返回值:cat <<EOF|ruby|(read ret;echo $ret):
linux@linuxidc: ruby_src$./read.sh
:98765432dcb
必须要加小括号哦,否则read ret和echo还是在2个进程,等于做无用功啊!你可以这样做:
linux@linuxidc: ruby_src$my_val=$(./read.sh)
linux@linuxidc: ruby_src$echo $my_val
:98765432dcb
这样就将返回结果保存在主bash进程中的my_val变量中了哦。或者你在while循环中自己处理吧,因为while循环和read也在一个进程里哦,改写后的shell代码如下:
#!/opt/local/bin/bash
export id="abc123456789" #只是示例,实际可以动态获取id的值
cat <<EOF | ruby | while read ret;do echo ">>>>" $ret;done
class String
def r_add_1
replace reverse.each_byte.map {|b|b+1}.pack("c*")
end
end
puts ENV["id"].dup.r_add_1
EOF
你要是觉得看的cat那一行太长,可以把do...done放到EOF结尾的后面,效果是一样的:
#!/opt/local/bin/bash
export id="abc123456789" #只是示例,实际可以动态获取id的值
cat <<EOF | ruby | while read ret
class String
def r_add_1
replace reverse.each_byte.map {|b|b+1}.pack("c*")
end
end
puts ENV["id"].dup.r_add_1
EOF
do
echo ">>>>" $ret
done
最后执行read.sh都会显示:
linux@linuxidc: ruby_src$./read.sh
>>>> :98765432dcb
一切皆有可能!你最后肯定还会问:为毛我不用$my_val=$(ruby -e"xxx")这种纯ruby的更简单方法,而要用shell+ruby混合的复杂方法呢??答案是:我不是在举ruby+shell的例子吗!?