Feature #15627
openAppearance of custom singleton classes
Description
When I have a singleton class AClass of an instance a of a custom class A,
class A; end a = A.new AClass = a.singleton_class i) even though the singleton class of nil, false, and true are referred to by their assigned constant names, the singleton class AClass of a is not:
nil.singleton_class #=> NilClass false.singleton_class #=> FalseClass true.singleton_class #=> TrueClass a.singleton_class #=> #<Class:#<A:0x00007fda832a7eb0>> ii) even though the singleton class of nil, false, and true appear as their class, the singleton class AClass of a does not:
nil.class #=> NilClass false.class #=> FalseClass true.class #=> TrueClass a.class #=> A This contrast between nil, false, and true on the one hand and a on the other is confusing. I am actually not sure if this is intended behaviour It may be related to
I expect AClass to behave the same as with NilClass, FalseClass, and TrueClass. I expect:
a.singleton_class #=> AClass a.class #=> AClass If the current behaviour is intended, I would like this to become a feature request.
Updated by Eregon (Benoit Daloze) over 6 years ago
singleton_class and class are different by design.
They are only the same for true, false and nil.
Having the singleton class get named when assigning it to a constant sounds like a possible feature.
Although it doesn't seem common to assign a singleton class to a constant.
Updated by nobu (Nobuyoshi Nakada) over 6 years ago
At first, as no syntax to name a singleton class like ordinary classes/modules, singleton classes cannot have a name.
And name-by-assignment is a “best effort” (or “better than nothing”).
Updated by mame (Yusuke Endoh) over 6 years ago
Rather, it looks a bug that #singleton_class returns a non-singleton class:
p Object.new.singleton_class.singleton_class? #=> true p true .singleton_class.singleton_class? #=> false p false.singleton_class.singleton_class? #=> false p nil .singleton_class.singleton_class? #=> false 1.singleton_class #=> can't define singleton (TypeError) It looks reasonable to raise an exception like 1.singleton_class. (But I'm unsure if it is worth enough to break compatibility.)
Updated by Hanmac (Hans Mackowiak) over 6 years ago
@mame (Yusuke Endoh) it is by design that true, false and nil has their class work as singleton class so you can do:
def true.bla # something end
Updated by mame (Yusuke Endoh) over 6 years ago
Wow.
def true.foo; end p TrueClass.instance_methods.include?(:foo) #=> true I didn't know, thanks. I have used Ruby for fifteen years, but Ruby still brings fresh surprise to me.
Updated by jeremyevans0 (Jeremy Evans) about 6 years ago
- Tracker changed from Bug to Feature
- Backport deleted (
2.4: UNKNOWN, 2.5: UNKNOWN, 2.6: UNKNOWN)
Updated by Anonymous over 5 years ago
That being said,
Why does TrueClass, FalseClass and NilClass classes in Object instead of just singleton_class of true, false and nil repectively?
If it’s for backwards compatibility, remember that with major updates (e.g. Ruby 3.0) we don’t need to guarantee backward-compatibility (Ruby 3’s being changes to how keyword arguments are parsed).
(true, false and nil are more like pseudo-constants compared to other pseudo-variables (self, __FILE__, __LINE__))