动态地将类解析为全局范围

时间:2018-06-19 07:25:54

标签: ruby-on-rails ruby class autoload

我需要将类的引用保存为哈希:

@hash['foo'] = bar if bar.is_a?(Class)

上面的代码存在于我的/lib目录中,每次都不会自动加载,bar通常是自动加载的类。为了避免在我reload!代码时出现“xxx的副本已从模块树中删除但仍处于活动状态”错误,我试图将bar解析为全局范围,即:在类名(::成为Baz)之前添加::Baz

我不确定如何动态执行此,不将类转换为字符串,前置::,然后将其转换回类。

1 个答案:

答案 0 :(得分:5)

在为散列分配常量时,常量在赋值时被解析:(不是特定于散列,这只是常量的工作原理)

hash = {}
A = 1
hash[:a] = A
#=> 1          # <- 1 is being assigned, not A

A = 2
hash[:a]
#=> 1

解决它的一种方法是存储常量的名称:

hash = {}
A = 1
hash[:a] = 'A'
#=> 'A'

并通过const_get / constantize解决问题:

A = 2
Object.const_get(hash[:a])
#=> 2

这也适用于嵌套常量:

hash[:pi] = 'Math::PI'
Object.const_get(hash[:pi])
#=> 3.141592653589793

如果您的对象恰好是命名类(或模块),则可以通过Module#name检索其名称:

hash[:lazy_enum] = Enumerator::Lazy.name
#=> "Enumerator::Lazy"

Object.const_get(hash[:lazy_enum])
#=> Enumerator::Lazy

另一种方法是使用proc引用其块中的常量:

hash = {}
A = 1
hash[:a] = -> { A }
#=> #<Proc:0x00007fc4ba05f510@(irb):10 (lambda)>

在调用块时将解析常量:

A = 2
hash[:a].call
#=> 2