如果使用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 ##
这个代码中,你应该能够用<tab>键来自动完成输入命令。现在让我们再进一步,添加目录自动完成。
例子:
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是知识海洋的贝壳,一起来捡吧。
高性价比培训,欢迎了解和交流。简学互动Ruby、Rails基础培训