为什么最顶层的 if 条件被忽略了?

Why are the top-most if conditions ignored?

我承认我对 Ruby 还是个新手,所以我还不熟悉其中的陷阱,仍在学习中。

我已经用谷歌搜索了很多次这个问题,但没能找到确切的答案。大多数结果谈论 "nested" if/else 语句。这不是我正在尝试的。我从 SO 找到了另一个关于根据条件映射数组的答案,但感觉这会让我回到我已经遇到的相同问题。

Link 到关于合并嵌套数组的 SO 文章,如果有人感兴趣的话:

我的问题:

在 ruby 中设计一个简单的 CLI 脚本时使用 optparse 来控制基于条件的输出。我遇到了一个问题,我无法执行多个顺序 if 语句和 concat/merge 将多个数组合并为一个以传递给另一个函数。

出于某种原因,只有最后一个 if 块 被兑现。在最后一个之前的所有 if 块 return nil.

任何人都可以告诉我我做错了什么并提供一些关于我的问题的相关文档。比如为什么会出现这样的问题?

希望有一个可行的解决方案可供学习,但参考 material 也可以,因为我的目标是从这个问题中学习。

我的测试结果:

$ ruby servers.rb
#=> ["server1", "server2", "server3"]
$ ruby servers.rb -m
#=> nil
$ ruby servers.rb -h
#=> nil
$ ruby servers.rb -s server6
#=> ["server6"]
$ ruby servers.rb -m -h
#=> nil

下面是我的脚本:

#!/usr/bin/env ruby
require 'optparse'

@servers = {
  'primary' => %w[server1 server2 server3],
  'inhouse' => %w[server4 server5],
  'mlm' => %w[server6]
}

@options = {
  inhouse: false,
  server: [],
  mlm: false
}

# Parse options passed to medusabackup.rb
optparse = OptionParser.new do |opts|
  opts.on('-h', '--inhouse', 'Limit results to inhouse results') do
    @options[:inhouse] = true
  end
  opts.on('-m', '--mlm', 'Include mlm results') do
    @options[:mlm] = true
  end
  opts.on('-s', '--server=SERVER', 'Limit results to SERVER') do |f|
    @options[:server] << f
  end
  opts.on_tail('--help', 'Display this screen') { puts opts, exit }
end

begin
  optparse.parse!
rescue OptionParser::InvalidOption, OptionParser::MissingArgument
  puts $ERROR_INFO.to_s
  puts optparse
  exit
end

def selected_servers
  # Setup Selected variable as an array
  selected = []
  # If @options[:server]  array is not empty and @options[:server]
  # add @options[:server] servers to [selected] array
  if @options[:server].empty?
    # If @options[:mlm] is true add @server['mlm'] servers to [selected] array
    selected.concat @servers['mlm'] if @options[:mlm]
    # If @options[:inhouse] is true add @server['inhouse'] servers to [selected] array
    selected.concat @servers['inhouse'] if @options[:inhouse]
    # If @options[:mlm], @options[:inhouse] is true add @server['mlm'] servers to [selected] array
    selected.concat @servers['primary'] unless @options[:mlm] || @options[:inhouse]
  else
    selected.concat @options[:server]
  end
end
puts selected_servers.inspect

感谢 Max 和大家指出我的错误。忘记return在函数底部选择。

selected_servers 中没有明确的 return,因此它 return 计算它运行的最后一个表达式的值,通常是失败的 unless。失败 if/unless returns nil.