我想了解以下代码的工作原理:
def url
@url ||= {
"basename" => self.basename,
"output_ext" => self.output_ext,
}.inject("/:basename/") { |result, token|
result.gsub(/:#{token.first}/, token.last)
}.gsub(/\/\//, "/")
end
我知道它的作用;不知何故,它返回对应于位于服务器上的dir的文件的URL。所以它返回类似于这样的字符串:/path/to/my/file.html
我了解如果@url
已有值,则会返回该值,并且会丢弃右||=
。我也明白这开始创建两个元素的哈希。
我也认为我理解最后的gsub;它用斜杠替换反斜杠(以应对Windows服务器,我猜)。
令我惊讶的是inject
部分。我无法理解它。我之前使用过inject
,但这个对我来说太过分了。我不知道如何使用each
,因为我不明白它是做什么的。
我为这个问题稍微修改了原来的功能;原文来自this jekyll file。
干杯!
答案 0 :(得分:2)
foo.inject(bar) {|result, x| f(result,x) }
始终可以写成:
result = bar
foo.each {|x| result = f(result, x)}
result
因此,对于您的情况,每个版本的版本如下所示:
result = "/:basename/"
{
"basename" => self.basename,
"output_ext" => self.output_ext,
}.each {|token|
result = result.gsub(/:#{token.first}/, token.last)
}
result
含义:对于散列中的所有键值对,"/:basename/"
中每次出现的键都将替换为值。
答案 1 :(得分:1)
也许拆分代码并调整一点帮助
options = { "basename" => self.basename, "output_ext" => self.output_ext }
options.inject("/:basename") do |result, key_and_kalue|
# Iterating over the hash yields an array of two elements, which I called key_and_value
result.gsub(":#{key_and_value[0]}", key_and_value[1])
end.gsub!(//\/\/, '/')
基本上,注入代码会迭代所有options
并替换实际值,只要它看到“:key”