gpt4 book ai didi

ruby - 为什么 Kernel#require 在 Ruby 中引发 LoadError?

转载 作者:数据小太阳 更新时间:2023-10-29 07:31:25 24 4
gpt4 key购买 nike

嗨,多年来我一直想知道为什么您不能使用 Kernel#require 方法来加载 gems。

例如,这将起作用:

#!/usr/bin/ruby -w
require 'ruby2d' # => true

这里require的所有者是Kernel:

p Object.method(:require).owner    # => Kernel
p Kernel.method(:require).owner # => #<Class:Kernel>

但这行得通:

p Object.send :require, 'ruby2d'    # => true
p String.send :require, 'ruby2d' # => false
p Kernel.require 'ruby2d' # => false

gem 'ruby2d'                        # => true
p String.send :require, 'ruby2d' # => true
p Kernel.require 'ruby2d' # => false

[坏主意,但你可以在任何对象上发送 require 方法]

不知何故这不起作用:

#!/usr/bin/ruby -w
p Kernel.require 'ruby2d'
Traceback (most recent call last):
1: from p.rb:2:in `<main>'
p.rb:2:in `require': cannot load such file -- ruby2d (LoadError)

这是怎么回事?

最佳答案

这里发生了几件事情,并以有趣的方式相互作用,我们需要拆开这些事情来了解正在发生的事情。

首先,require 是如何工作的。有一个包含目录列表的全局变量 $LOAD_PATHrequire 工作的“原始”方式(即没有 Rubygems)是 Ruby 将简单地搜索此列表以查找所需文件,如果找到则加载它,否则会引发异常。

Rubygems 改变了这一点。加载 Rubygems 时 replaces the built-in require method with its own , 首先给原来的别名。这个新方法首先调用原始方法,如果没有找到所需的文件,那么它不会立即引发异常,而是搜索已安装的 gem,如果找到匹配的文件,则该 gem 激活。这意味着(除其他外)gem 的 lib 目录被添加到 $LOAD_PATH

即使 Rubygems 现在是 Ruby 的一部分并默认安装,它仍然是一个单独的库,原始代码仍然存在。 (您可以使用 --disable=gems 禁用加载 Rubygems)。

接下来,我们可以看看原来的require方法是如何定义的。是done with the C function rb_define_global_function .这个函数依次calls rb_define_module_function ,以及那个函数 looks like :

void
rb_define_module_function(VALUE module, const char *name, VALUE (*func)(ANYARGS), int argc)
{
rb_define_private_method(module, name, func, argc);
rb_define_singleton_method(module, name, func, argc);
}

如您所见,该方法最终被定义两次,一次作为私有(private)方法(即包含在 Object 中并随处可用的方法),一次作为 Kernel 上的单例方法(即类方法)。

现在我们可以开始看看发生了什么。 Rubygems 代码仅替换包含的 require 版本。当您调用 Kernel.require 时,您会得到原始的 require 方法,它对 Rubygems 一无所知。

如果你跑

p Kernel.require 'ruby2d'

您将遇到与在禁用 Rubygems (ruby --disable=gems p.rb) 时运行以下命令相同的错误:

p require 'ruby2d'

在这两种情况下我得到:

Traceback (most recent call last):
1: from p.rb:1:in `<main>'
p.rb:1:in `require': cannot load such file -- ruby2d (LoadError)

这与我运行第二个示例 Rubygems 不同,在这种情况下我得到(因为我没有安装 gem):

Traceback (most recent call last):
2: from p.rb:1:in `<main>'
1: from /Users/matt/.rubies/ruby-2.6.1/lib/ruby/2.6.0/rubygems/core_ext/kernel_require.rb:54:in `require'
/Users/matt/.rubies/ruby-2.6.1/lib/ruby/2.6.0/rubygems/core_ext/kernel_require.rb:54:in `require': cannot load such file -- ruby2d (LoadError)

两个都是 LoadError,但是一个已经通过 Rubygems,一个还没有。

Kernel.require 似乎可以工作的示例也可以解释,因为在那些情况下文件已经加载,原始的 require 代码只是看到一个已加载文件并返回 false。 Kernel.require 也可以工作的另一个例子是

gem 'ruby2d'
Kernel.require 'ruby2d'

gem 方法激活 gem,尽管它不加载它。如上所述,这会将 gems lib 目录(包含作为 require 目标的文件)添加到 $LOAD_PATH,因此原始 require 代码将找到它并加载它。

关于ruby - 为什么 Kernel#require 在 Ruby 中引发 LoadError?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57235261/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com