class GetoptLong
GetoptLong 类为选项和常规参数提供了解析功能。
使用 GetoptLong,您可以为程序定义选项。然后,程序可以捕获并响应执行该程序的命令中包含的任何选项。
一个简单的例子:文件 simple.rb
require 'getoptlong' options = GetoptLong.new( ['--number', '-n', GetoptLong::REQUIRED_ARGUMENT], ['--verbose', '-v', GetoptLong::OPTIONAL_ARGUMENT], ['--help', '-h', GetoptLong::NO_ARGUMENT] )
如果您对选项有些熟悉,您可能想跳到这个完整示例。
选项¶ ↑
一个 GetoptLong 选项具有
-
一个字符串选项名称。
-
零个或多个字符串的名称别名。
-
一个选项类型。
可以通过调用单例方法 GetoptLong.new
来定义选项,该方法返回一个新的 GetoptLong 对象。然后,可以通过调用其他方法(如 GetoptLong#each
)来处理选项。
选项名称和别名¶ ↑
在定义选项的数组中,第一个元素是字符串选项名称。通常,名称采用“长”形式,以两个连字符开头。
选项名称可以有任意数量的别名,这些别名由附加的字符串元素定义。
名称和每个别名必须是以下两种形式之一
-
两个连字符,后跟一个或多个字母。
-
一个连字符,后跟一个字母。
文件 aliases.rb
require 'getoptlong' options = GetoptLong.new( ['--xxx', '-x', '--aaa', '-a', '-p', GetoptLong::NO_ARGUMENT] ) options.each do |option, argument| p [option, argument] end
可以通过其名称或任何别名引用选项;解析的选项始终报告名称,而不是别名
$ ruby aliases.rb -a -p --xxx --aaa -x
输出
["--xxx", ""] ["--xxx", ""] ["--xxx", ""] ["--xxx", ""] ["--xxx", ""]
也可以通过其名称或任何别名的缩写形式来引用选项,只要该缩写在选项中是唯一的。
文件 abbrev.rb
require 'getoptlong' options = GetoptLong.new( ['--xxx', GetoptLong::NO_ARGUMENT], ['--xyz', GetoptLong::NO_ARGUMENT] ) options.each do |option, argument| p [option, argument] end
命令行
$ ruby abbrev.rb --xxx --xx --xyz --xy
输出
["--xxx", ""] ["--xxx", ""] ["--xyz", ""] ["--xyz", ""]
此命令行引发 GetoptLong::AmbiguousOption
$ ruby abbrev.rb --x
重复¶ ↑
一个选项可以被引用多次
$ ruby abbrev.rb --xxx --xyz --xxx --xyz
输出
["--xxx", ""] ["--xyz", ""] ["--xxx", ""] ["--xyz", ""]
将剩余选项视为参数¶ ↑
在标记 --
之后出现的任何位置的类似选项的标记都被视为普通参数,而不是作为选项进行处理
$ ruby abbrev.rb --xxx --xyz -- --xxx --xyz
输出
["--xxx", ""] ["--xyz", ""]
选项类型¶ ↑
每个选项定义都包含一个选项类型,该类型控制选项是否需要参数。
文件 types.rb
require 'getoptlong' options = GetoptLong.new( ['--xxx', GetoptLong::REQUIRED_ARGUMENT], ['--yyy', GetoptLong::OPTIONAL_ARGUMENT], ['--zzz', GetoptLong::NO_ARGUMENT] ) options.each do |option, argument| p [option, argument] end
请注意,选项类型与选项参数(它是必需的、可选的还是禁止的)有关,而不是与选项本身是否是必需的有关。
带必需参数的选项¶ ↑
类型为 GetoptLong::REQUIRED_ARGUMENT
的选项必须后跟一个参数,该参数与该选项关联
$ ruby types.rb --xxx foo
输出
["--xxx", "foo"]
如果该选项不是最后一个,则其参数是它后面的任何内容(即使该参数看起来像另一个选项)
$ ruby types.rb --xxx --yyy
输出
["--xxx", "--yyy"]
如果该选项是最后一个,则会引发异常
$ ruby types.rb # Raises GetoptLong::MissingArgument
带可选参数的选项¶ ↑
类型为 GetoptLong::OPTIONAL_ARGUMENT
的选项可以后跟一个参数,如果给定,则该参数与该选项关联。
如果该选项是最后一个,则它没有参数
$ ruby types.rb --yyy
输出
["--yyy", ""]
如果该选项后跟另一个选项,则它没有参数
$ ruby types.rb --yyy --zzz
输出
["--yyy", ""] ["--zzz", ""]
否则,该选项后跟其参数,该参数与该选项关联
$ ruby types.rb --yyy foo
输出
["--yyy", "foo"]
无参数的选项¶ ↑
类型为 GetoptLong::NO_ARGUMENT
的选项不带参数
ruby types.rb --zzz foo
输出
["--zzz", ""]
ARGV¶ ↑
您可以使用方法 each
和一个代码块,或者使用方法 get
来处理选项。
在处理过程中,找到的每个选项都会被删除,以及它的参数(如果有)。处理后,每个剩余元素都不是选项,也不是选项的参数。
文件 argv.rb
require 'getoptlong' options = GetoptLong.new( ['--xxx', GetoptLong::REQUIRED_ARGUMENT], ['--yyy', GetoptLong::OPTIONAL_ARGUMENT], ['--zzz', GetoptLong::NO_ARGUMENT] ) puts "Original ARGV: #{ARGV}" options.each do |option, argument| p [option, argument] end puts "Remaining ARGV: #{ARGV}"
命令行
$ ruby argv.rb --xxx Foo --yyy Bar Baz --zzz Bat Bam
输出
Original ARGV: ["--xxx", "Foo", "--yyy", "Bar", "Baz", "--zzz", "Bat", "Bam"] ["--xxx", "Foo"] ["--yyy", "Bar"] ["--zzz", ""] Remaining ARGV: ["Baz", "Bat", "Bam"]
排序¶ ↑
有三个设置控制选项的解释方式
-
PERMUTE
. -
REQUIRE_ORDER
. -
RETURN_IN_ORDER
.
如果定义了环境变量 POSIXLY_CORRECT
,则新的 GetoptLong 对象的初始设置为 REQUIRE_ORDER
,否则为 PERMUTE
。
PERMUTE 排序¶ ↑
在 PERMUTE
排序中,选项和其他非选项参数可以以任何顺序和任何混合形式出现。
文件 permute.rb
require 'getoptlong' options = GetoptLong.new( ['--xxx', GetoptLong::REQUIRED_ARGUMENT], ['--yyy', GetoptLong::OPTIONAL_ARGUMENT], ['--zzz', GetoptLong::NO_ARGUMENT] ) puts "Original ARGV: #{ARGV}" options.each do |option, argument| p [option, argument] end puts "Remaining ARGV: #{ARGV}"
命令行
$ ruby permute.rb Foo --zzz Bar --xxx Baz --yyy Bat Bam --xxx Bag Bah
输出
Original ARGV: ["Foo", "--zzz", "Bar", "--xxx", "Baz", "--yyy", "Bat", "Bam", "--xxx", "Bag", "Bah"] ["--zzz", ""] ["--xxx", "Baz"] ["--yyy", "Bat"] ["--xxx", "Bag"] Remaining ARGV: ["Foo", "Bar", "Bam", "Bah"]
REQUIRE_ORDER 排序¶ ↑
在 REQUIRE_ORDER
排序中,所有选项都位于所有非选项之前;也就是说,第一个非选项单词之后的每个单词都被视为非选项单词(即使它以连字符开头)。
文件 require_order.rb
require 'getoptlong' options = GetoptLong.new( ['--xxx', GetoptLong::REQUIRED_ARGUMENT], ['--yyy', GetoptLong::OPTIONAL_ARGUMENT], ['--zzz', GetoptLong::NO_ARGUMENT] ) options.ordering = GetoptLong::REQUIRE_ORDER puts "Original ARGV: #{ARGV}" options.each do |option, argument| p [option, argument] end puts "Remaining ARGV: #{ARGV}"
命令行
$ ruby require_order.rb --xxx Foo Bar --xxx Baz --yyy Bat -zzz
输出
Original ARGV: ["--xxx", "Foo", "Bar", "--xxx", "Baz", "--yyy", "Bat", "-zzz"] ["--xxx", "Foo"] Remaining ARGV: ["Bar", "--xxx", "Baz", "--yyy", "Bat", "-zzz"]
RETURN_IN_ORDER 排序¶ ↑
在 RETURN_IN_ORDER
排序中,每个单词都被视为一个选项。以一个连字符(或两个)开头的单词以通常的方式处理;一个不以这种方式开头的单词 word
被视为一个名称为空字符串的选项,其值为 word
。
文件 return_in_order.rb
require 'getoptlong' options = GetoptLong.new( ['--xxx', GetoptLong::REQUIRED_ARGUMENT], ['--yyy', GetoptLong::OPTIONAL_ARGUMENT], ['--zzz', GetoptLong::NO_ARGUMENT] ) options.ordering = GetoptLong::RETURN_IN_ORDER puts "Original ARGV: #{ARGV}" options.each do |option, argument| p [option, argument] end puts "Remaining ARGV: #{ARGV}"
命令行
$ ruby return_in_order.rb Foo --xxx Bar Baz --zzz Bat Bam
输出
Original ARGV: ["Foo", "--xxx", "Bar", "Baz", "--zzz", "Bat", "Bam"] ["", "Foo"] ["--xxx", "Bar"] ["", "Baz"] ["--zzz", ""] ["", "Bat"] ["", "Bam"] Remaining ARGV: []
完整示例¶ ↑
文件 fibonacci.rb
require 'getoptlong' options = GetoptLong.new( ['--number', '-n', GetoptLong::REQUIRED_ARGUMENT], ['--verbose', '-v', GetoptLong::OPTIONAL_ARGUMENT], ['--help', '-h', GetoptLong::NO_ARGUMENT] ) def help(status = 0) puts <<~HELP Usage: -n n, --number n: Compute Fibonacci number for n. -v [boolean], --verbose [boolean]: Show intermediate results; default is 'false'. -h, --help: Show this help. HELP exit(status) end def print_fibonacci (number) return 0 if number == 0 return 1 if number == 1 or number == 2 i = 0 j = 1 (2..number).each do k = i + j i = j j = k puts j if @verbose end puts j unless @verbose end options.each do |option, argument| case option when '--number' @number = argument.to_i when '--verbose' @verbose = if argument.empty? true elsif argument.match(/true/i) true elsif argument.match(/false/i) false else puts '--verbose argument must be true or false' help(255) end when '--help' help end end unless @number puts 'Option --number is required.' help(255) end print_fibonacci(@number)
命令行
$ ruby fibonacci.rb
输出
Option --number is required. Usage: -n n, --number n: Compute Fibonacci number for n. -v [boolean], --verbose [boolean]: Show intermediate results; default is 'false'. -h, --help: Show this help.
命令行
$ ruby fibonacci.rb --number
引发 GetoptLong::MissingArgument
fibonacci.rb: option `--number' requires an argument
命令行
$ ruby fibonacci.rb --number 6
输出
8
命令行
$ ruby fibonacci.rb --number 6 --verbose
输出
1 2 3 5 8
命令行
$ ruby fibonacci.rb --number 6 --verbose yes
输出
--verbose argument must be true or false Usage: -n n, --number n: Compute Fibonacci number for n. -v [boolean], --verbose [boolean]: Show intermediate results; default is 'false'. -h, --help: Show this help.
常量
- ARGUMENT_FLAGS
参数标志。
- ORDERINGS
排序。
- STATUS_TERMINATED
- VERSION
版本。
属性
返回选项处理是否失败。
返回选项处理是否失败。
返回排序设置。
设置静默模式并返回给定的参数
-
当
false
或nil
时,错误消息将写入$stdout
。 -
否则,不会写入错误消息。
设置静默模式并返回给定的参数
-
当
false
或nil
时,错误消息将写入$stdout
。 -
否则,不会写入错误消息。
公共类方法
返回一个基于给定 arguments
的新 GetoptLong 对象。请参阅选项。
示例
require 'getoptlong' options = GetoptLong.new( ['--number', '-n', GetoptLong::REQUIRED_ARGUMENT], ['--verbose', '-v', GetoptLong::OPTIONAL_ARGUMENT], ['--help', '-h', GetoptLong::NO_ARGUMENT] )
如果以下情况,则引发异常
-
arguments
的任何一个都不是数组。 -
任何选项名称或别名都不是字符串。
-
任何选项类型无效。
# File getoptlong-0.2.1/lib/getoptlong.rb, line 412 def initialize(*arguments) # # Current ordering. # if ENV.include?('POSIXLY_CORRECT') @ordering = REQUIRE_ORDER else @ordering = PERMUTE end # # Hash table of option names. # Keys of the table are option names, and their values are canonical # names of the options. # @canonical_names = Hash.new # # Hash table of argument flags. # Keys of the table are option names, and their values are argument # flags of the options. # @argument_flags = Hash.new # # Whether error messages are output to $stderr. # @quiet = false # # Status code. # @status = STATUS_YET # # Error code. # @error = nil # # Error message. # @error_message = nil # # Rest of catenated short options. # @rest_singles = '' # # List of non-option-arguments. # Append them to ARGV when option processing is terminated. # @non_option_arguments = Array.new if 0 < arguments.length set_options(*arguments) end end
公共实例方法
使用每个选项调用给定的代码块;每个选项都是一个包含以下内容的 2 元素数组
-
选项名称(名称本身,而不是别名)。
-
选项值。
示例
require 'getoptlong' options = GetoptLong.new( ['--xxx', '-x', GetoptLong::REQUIRED_ARGUMENT], ['--yyy', '-y', GetoptLong::OPTIONAL_ARGUMENT], ['--zzz', '-z',GetoptLong::NO_ARGUMENT] ) puts "Original ARGV: #{ARGV}" options.each do |option, argument| p [option, argument] end puts "Remaining ARGV: #{ARGV}"
命令行
ruby each.rb -xxx Foo -x Bar --yyy Baz -y Bat --zzz
输出
Original ARGV: ["-xxx", "Foo", "-x", "Bar", "--yyy", "Baz", "-y", "Bat", "--zzz"] ["--xxx", "xx"] ["--xxx", "Bar"] ["--yyy", "Baz"] ["--yyy", "Bat"] ["--zzz", ""] Remaining ARGV: ["Foo"]
# File getoptlong-0.2.1/lib/getoptlong.rb, line 859 def each loop do option_name, option_argument = get_option break if option_name == nil yield option_name, option_argument end end
返回 POSIX 定义格式的相应错误消息。如果没有发生错误,则返回 nil
。
# File getoptlong-0.2.1/lib/getoptlong.rb, line 662 def error_message return @error_message end
以 2 元素数组的形式返回下一个选项,其中包含
-
选项名称(名称本身,而不是别名)。
-
选项值。
如果没有更多选项,则返回 nil
。
# File getoptlong-0.2.1/lib/getoptlong.rb, line 674 def get option_name, option_argument = nil, '' # # Check status. # return nil if @error != nil case @status when STATUS_YET @status = STATUS_STARTED when STATUS_TERMINATED return nil end # # Get next option argument. # if 0 < @rest_singles.length argument = '-' + @rest_singles elsif (ARGV.length == 0) terminate return nil elsif @ordering == PERMUTE while 0 < ARGV.length && ARGV[0] !~ /\A-./ @non_option_arguments.push(ARGV.shift) end if ARGV.length == 0 terminate return nil end argument = ARGV.shift elsif @ordering == REQUIRE_ORDER if (ARGV[0] !~ /\A-./) terminate return nil end argument = ARGV.shift else argument = ARGV.shift end # # Check the special argument `--'. # `--' indicates the end of the option list. # if argument == '--' && @rest_singles.length == 0 terminate return nil end # # Check for long and short options. # if argument =~ /\A(--[^=]+)/ && @rest_singles.length == 0 # # This is a long style option, which start with `--'. # pattern = $1 if @canonical_names.include?(pattern) option_name = pattern else # # The option `option_name' is not registered in `@canonical_names'. # It may be an abbreviated. # matches = [] @canonical_names.each_key do |key| if key.index(pattern) == 0 option_name = key matches << key end end if 2 <= matches.length set_error(AmbiguousOption, "option `#{argument}' is ambiguous between #{matches.join(', ')}") elsif matches.length == 0 set_error(InvalidOption, "unrecognized option `#{argument}'") end end # # Check an argument to the option. # if @argument_flags[option_name] == REQUIRED_ARGUMENT if argument =~ /=(.*)/m option_argument = $1 elsif 0 < ARGV.length option_argument = ARGV.shift else set_error(MissingArgument, "option `#{argument}' requires an argument") end elsif @argument_flags[option_name] == OPTIONAL_ARGUMENT if argument =~ /=(.*)/m option_argument = $1 elsif 0 < ARGV.length && ARGV[0] !~ /\A-./ option_argument = ARGV.shift else option_argument = '' end elsif argument =~ /=(.*)/m set_error(NeedlessArgument, "option `#{option_name}' doesn't allow an argument") end elsif argument =~ /\A(-(.))(.*)/m # # This is a short style option, which start with `-' (not `--'). # Short options may be catenated (e.g. `-l -g' is equivalent to # `-lg'). # option_name, ch, @rest_singles = $1, $2, $3 if @canonical_names.include?(option_name) # # The option `option_name' is found in `@canonical_names'. # Check its argument. # if @argument_flags[option_name] == REQUIRED_ARGUMENT if 0 < @rest_singles.length option_argument = @rest_singles @rest_singles = '' elsif 0 < ARGV.length option_argument = ARGV.shift else # 1003.2 specifies the format of this message. set_error(MissingArgument, "option requires an argument -- #{ch}") end elsif @argument_flags[option_name] == OPTIONAL_ARGUMENT if 0 < @rest_singles.length option_argument = @rest_singles @rest_singles = '' elsif 0 < ARGV.length && ARGV[0] !~ /\A-./ option_argument = ARGV.shift else option_argument = '' end end else # # This is an invalid option. # 1003.2 specifies the format of this message. # if ENV.include?('POSIXLY_CORRECT') set_error(InvalidOption, "invalid option -- #{ch}") else set_error(InvalidOption, "invalid option -- #{ch}") end end else # # This is a non-option argument. # Only RETURN_IN_ORDER fell into here. # return '', argument end return @canonical_names[option_name], option_argument end
设置排序;请参阅排序;返回新的排序。
如果给定的 ordering
是 PERMUTE
并且定义了环境变量 POSIXLY_CORRECT
,则将排序设置为 REQUIRE_ORDER
;否则将排序设置为 ordering
options = GetoptLong.new options.ordering == GetoptLong::PERMUTE # => true options.ordering = GetoptLong::RETURN_IN_ORDER options.ordering == GetoptLong::RETURN_IN_ORDER # => true ENV['POSIXLY_CORRECT'] = 'true' options.ordering = GetoptLong::PERMUTE options.ordering == GetoptLong::REQUIRE_ORDER # => true
如果 ordering
无效,则引发异常。
# File getoptlong-0.2.1/lib/getoptlong.rb, line 489 def ordering=(ordering) # # The method is failed if option processing has already started. # if @status != STATUS_YET set_error(ArgumentError, "argument error") raise RuntimeError, "invoke ordering=, but option processing has already started" end # # Check ordering. # if !ORDERINGS.include?(ordering) raise ArgumentError, "invalid ordering `#{ordering}'" end if ordering == PERMUTE && ENV.include?('POSIXLY_CORRECT') @ordering = REQUIRE_ORDER else @ordering = ordering end end
将现有选项替换为 arguments
给定的选项,该选项具有与 ::new
的参数相同的形式;返回 self
。
如果选项处理已开始,则引发异常。
# File getoptlong-0.2.1/lib/getoptlong.rb, line 524 def set_options(*arguments) # # The method is failed if option processing has already started. # if @status != STATUS_YET raise RuntimeError, "invoke set_options, but option processing has already started" end # # Clear tables of option names and argument flags. # @canonical_names.clear @argument_flags.clear arguments.each do |arg| if !arg.is_a?(Array) raise ArgumentError, "the option list contains non-Array argument" end # # Find an argument flag and it set to `argument_flag'. # argument_flag = nil arg.each do |i| if ARGUMENT_FLAGS.include?(i) if argument_flag != nil raise ArgumentError, "too many argument-flags" end argument_flag = i end end raise ArgumentError, "no argument-flag" if argument_flag == nil canonical_name = nil arg.each do |i| # # Check an option name. # next if i == argument_flag begin if !i.is_a?(String) || i !~ /\A-([^-]|-.+)\z/ raise ArgumentError, "an invalid option `#{i}'" end if (@canonical_names.include?(i)) raise ArgumentError, "option redefined `#{i}'" end rescue @canonical_names.clear @argument_flags.clear raise end # # Register the option (`i') to the `@canonical_names' and # `@canonical_names' Hashes. # if canonical_name == nil canonical_name = i end @canonical_names[i] = canonical_name @argument_flags[i] = argument_flag end raise ArgumentError, "no option name" if canonical_name == nil end return self end
终止选项处理;如果处理已终止,则返回 nil
;否则返回 self
。
# File getoptlong-0.2.1/lib/getoptlong.rb, line 612 def terminate return nil if @status == STATUS_TERMINATED raise RuntimeError, "an error has occurred" if @error != nil @status = STATUS_TERMINATED @non_option_arguments.reverse_each do |argument| ARGV.unshift(argument) end @canonical_names = nil @argument_flags = nil @rest_singles = nil @non_option_arguments = nil return self end
如果选项处理已终止,则返回 true
,否则返回 false
。
# File getoptlong-0.2.1/lib/getoptlong.rb, line 632 def terminated? return @status == STATUS_TERMINATED end
受保护的实例方法
设置一个错误(一个受保护的方法)。
# File getoptlong-0.2.1/lib/getoptlong.rb, line 639 def set_error(type, message) $stderr.print("#{$0}: #{message}\n") if !@quiet @error = type @error_message = message @canonical_names = nil @argument_flags = nil @rest_singles = nil @non_option_arguments = nil raise type, message end