foz*_*foz 38
还有:
Gem.available?('somegem')
Run Code Online (Sandbox Code Playgroud)
您也可以使用正则表达式.如果我想允许'rcov'和GitHub变体如'relevant-rcov',我会很方便:
Gem.available?(/-?rcov$/)
Run Code Online (Sandbox Code Playgroud)
Eri*_*sel 35
查看Gem API文档,使用Gem :: Specification :: find_all_by_name来测试gem的可用性似乎是合理的.
if Gem::Specification::find_all_by_name('gemname').any?
do stuff
end
Run Code Online (Sandbox Code Playgroud)
find_all_by_name总是返回一个(规范对象)数组,相反,find_by_name如果没有找到匹配则引发异常.
Sim*_*tti 31
恕我直言,最好的方法是尝试加载/要求GEM并拯救Exception,正如Ray已经表明的那样.抢救LoadError异常是安全的,因为它不是由GEM本身引发的,而是require命令的标准行为.
您也可以使用gem命令.
begin
gem "somegem"
# with requirements
gem "somegem", ">=2.0"
rescue Gem::LoadError
# not installed
end
Run Code Online (Sandbox Code Playgroud)
gem命令与require命令具有相同的行为,但略有不同.AFAIK,它仍然试图自动加载主GEM文件.
深入研究rubygems.rb文件(第310行),我发现了以下执行
matches = Gem.source_index.find_name(gem.name, gem.version_requirements)
report_activate_error(gem) if matches.empty?
Run Code Online (Sandbox Code Playgroud)
它可以为您提供有关如何在不实际加载库的情况下进行脏检查的一些提示.
foz*_*foz 29
Gem.available?不赞成(唉!),你必须再次营救(双aaargh).是的,如果找不到gem,find_by_name会抛出异常.因此,为了与旧的rubygems向后兼容,常见的解决方案似乎是:
def gem_available?(name)
Gem::Specification.find_by_name(name)
rescue Gem::LoadError
false
rescue
Gem.available?(name)
end
Run Code Online (Sandbox Code Playgroud)
请注意,新方法允许您传递特定版本以查看是否已加载:
Gem::Specification.find_by_name('rails', '3.0.4')
Run Code Online (Sandbox Code Playgroud)
你可以:
begin
require "somegem"
rescue LoadError
# not installed
end
Run Code Online (Sandbox Code Playgroud)
但是,这不会告诉您模块是通过 gem 还是其他方式安装的。