def class A
def a
raise "hi" #can't be reached
end
class B
def b
a() #doesn't find method a.
end
end
end
I want to invoke a from b and raise the exception. How can I?
def class A
def a
raise "hi" #can't be reached
end
class B
def b
a() #doesn't find method a.
end
end
end
I want to invoke a from b and raise the exception. How can I?
Ruby doesn't have nested classes.
The only way to inherit behavior is, well, via inheritance.
If you want your code to work, you need to use a language which supports nested classes. While this is an incredibly neat and powerful feature, I unfortunately know of only two languages that have nested classes:
I don't know of any other.
Java has a construct called nested classes, but they have some unfortunate design limitations.
In your example above, it's not the class
B
that is nested insideA
, it is the constantB
that is nested insideA
. Think about this:Now, the class is available under two names:
A::B
andC
. It should be immediately obvious thatC
is global and not nested insideA
. (Well, actually,C
is nested insideObject
, because there aren't really global constants either, but that's beside the point.) But sinceC
andA::B
are the same class, it obviously cannot be both nested and not nested. The only logical conclusion is that the class itself isn't nested.The defining feature of nested classes is that method lookup goes along two dimensions: up the inheritance chain, and outwards through the nesting. Ruby, like 99.9% of all OO languages, only supports the former. (In some sense, nested classes inherit not only the features of their superclass, but also the features of their surrounding class.)
If you want then nested class to extend the outer class, then do so:
Was
a
supposed to be a class method for classA
?If you want to keep it as an instance method, you'll obviously have call to it on an instance, like for example
A.new.a
.Well depending on your circumstances there is actually a solution, a pretty easy one at that. Ruby allows the catching of method calls that aren't captured by the object. So for your example you could do:
So then if you do this:
you get:
It is probably an easier way to make something like a SubController that only handles certain activities, but can easily call basic controller methods (You would want to send in the parent controller as an argument in the initializer though).
Obviously this should be used sparingly, and it can really get confusing if you use it all over the place, but it can be really great to simplify your code.
I typically do something like this:
This is just for the lulz: