linux中直接运行ruby代码,25行Ruby代码编写Shell(译)

如果使用Linux或Mac,每次你打开一个终端就在使用shell应用程序。shell是一个接口,帮助你在你的系统执行命令。

此外,外壳也提供环境变量和有用的功能,如一个命令历史和自动完成。

如果你是喜欢深入学习的人,这篇文章将十分适合你!

一个Shell是如何工作的呢?

为了构建自己的shell应用程序,让我们思考一个shell是什么:第一,有一个提示,通常与一些额外的信息,比如你的当前用户和当前目录,然后输入一个命令,当你按下enter结果都显示在你的屏幕上。

是的,这听起来非常基础,但是这没有提醒你什么吗?

如果你想到 pry那么你是对的! shell基本上就是适用于您的操作系统的REPL(Read-Eval-Print-Loop)。

了解这些后,我们可以写您的shell的第一个版本:

prompt = "> "

print prompt

while (input = gets.chomp)

break if input == "exit"

system(input)

print prompt

end

这将给我们一个最小,但能工作的shell。我们可以通过使用许多其他类似应用程序使用的库来改善。那个库被称为 Readline.

使用Readline库

Readline是Ruby标准库的一部分,所以不用安装,你只需要 require它。

使用Readline的优势之一是,它可以自动保持命令历史。它还可以照顾打印命令提示符和许多其他的事情。

这里是我们shell的v2,这次使用了Readline:

require 'readline'

while input = Readline.readline("> ", true)

break if input == "exit"

system(input)

end

太好了,我们摆脱了 puts提示,而且现在我们有一些Readline的强大的功能。例如,我们可以使用键盘快捷键来删除一个字(CTRL + W),甚至搜索历史(CTRL + R)!

让我们添加一个新的命令打印完整的历史:

require 'readline'

while input = Readline.readline("> ", true)

break if input == "exit"

puts Readline::HISTORY.to_a if input == "hist"

# Remove blank lines from history

Readline::HISTORY.pop if input == ""

system(input)

end

有趣的事实:如果你用pry尝试这段代码,你会看到pry的命令历史记录!原因是pry也使用 Readline, 而Readline::HISTORY用来共享状态。

现在你可以输入 hist 获取命令历史记录:)

添加自动完成

谢谢这个你喜欢的外壳的自动完成功能帮你少打许多字。Readline很容易地将此功能集成到您的shell。

让我们从我们的命令历史来实现自动完成命令。

例子:

comp = proc { |s| LIST.grep(/^#{Regexp.escape(s)}/) }

Readline.completion_append_character = " "

Readline.completion_proc = comp

## rest of the code goes here ##

这个代码中,你应该能够用键来自动完成输入命令。现在让我们再进一步,添加目录自动完成。

例子:

comp = proc do |s|

directory_list = Dir.glob("#{s}*")

if directory_list.size > 0

directory_list

else

Readline::HISTORY.grep(/^#{Regexp.escape(s)}/)

end

end

completion_proc返回列表可能的候选人,在这种情况下,我们只需要用Dir.glob检查输入的字符串是否使用目录名称的一部分。Readline将处理其余的事情!

系统实现方法

现在你已经有一个不太差的外壳,带有命令历史&自动完成,只有25行代码:)

但是总有一些事情我想深入挖掘,所以你可以看到实际执行一个命令的时候,在幕后发生了什么。

这是用的system方法,这种方法在C语言只是发送你的命令到/bin/sh,这是一个shell应用程序。让我们来看看你可以在Ruby中实现/bin/sh所做的。

注意:这个只会工作在Linux / Mac:)

system方法

def system(command)

fork {

exec(command)

}

end

这里发生的是, fork为当前进程创建一个新的副本,然后通过 exec方法,这个进程被我们要运行的命令替换。这在Linux编程是一个很常见的模式。

如果你不fork,那么当前进程将被取代,这意味着当您正在运行的命令(ls, cd或其他)完成,你的Ruby程序将终止。

你可以看到这里发生的:

def system(command)

exec(command)

end

system('ls')

# This code will never run!

puts "after system"

结论

在本帖中,你知道一个shell是一个添加的与系统进行交互的接口(想想 irb / pry)。您还了解了如何通过使用强大的Readline库构建自己的壳,它提供了许多内置功能,如历史&自动完成(但你必须定义如何工作)。

然后您学习了 fork+ exec模式,通常用于Linux编程项目。

如果你喜欢这篇文章你能帮我一个忙分享与你所有的喜欢Ruby的朋友吗?它将帮助博客成长和更多的人可以学习:)

广告时间:

Ruby是知识海洋的贝壳,一起来捡吧。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值