首页 > 解决方案 > 将类动态解析为全局范围

问题描述

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

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

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

我不确定如何在将类转换为字符串、预先添加::、然后将其转换回类的情况下动态执行此操作。

标签: ruby-on-railsrubyclassautoload

解决方案


将常量分配给哈希时,常量在分配时被解析:(不是特定于哈希,这就是常量的工作方式)

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

推荐阅读