Super super ?

  • Thread starter Marc-antoine Kruzik
  • Start date
M

Marc-antoine Kruzik

Hello, I'm looking for informations about "super".

I have 3 classes : grandfather, father, and son.
And they all have a method named "say_hello".



Code:
class Grand_father
def say_hello
puts "Grand Father : Hello"
end
end

class Father < Grand_father
def say_hello
super
puts "Father : Hello"
end
end

class Son < Father
def say_hello
super
puts "Son : Hello"
end
end

s = Son.new
s.say_hello
Grand Father : Hello
Father : Hello
Son : Hello

But now, I would that Son didn't call Father, but directly Grand_father.

s.say_hello
Grand Father : Hello
Son : Hello

How could I do that ?
 
I

Ilan Berci

Marc-antoine Kruzik said:
But now, I would that Son didn't call Father, but directly Grand_father.

s.say_hello
Grand Father : Hello
Son : Hello

How could I do that ?

I am taking a break at work so I can't test it, but could you please
try:
super.super.say_hello
 
R

Robert Klemme

Hello, I'm looking for informations about "super".

I have 3 classes : grandfather, father, and son.
And they all have a method named "say_hello".



Code:
class Grand_father
def say_hello
puts "Grand Father : Hello"
end
end

class Father < Grand_father
def say_hello
super
puts "Father : Hello"
end
end

class Son < Father
def say_hello
super
puts "Son : Hello"
end
end

s = Son.new
s.say_hello
Grand Father : Hello
Father : Hello
Son : Hello

But now, I would that Son didn't call Father, but directly Grand_father.

s.say_hello
Grand Father : Hello
Son : Hello

How could I do that ?

You can do

class Son < Father
def say_hello
Grand_father.instance_method:)say_hello).bind(self).call
puts "Son : Hello"
end
end

However, as the weird construction indicates something might be wrong
with this kind of design. If you assume that the method actually had
side effects on the instance's state and you leave out one class in the
chain you likely end up with an instance in an inconsistent state. In
that case the design probably needs some work.

Btw, if Father does not define say_hello super will automatically call
Grand_father's method.

Kind regards

robert
 
R

Robert Klemme

I am taking a break at work so I can't test it, but could you please
try:
super.super.say_hello

This won't work - at least not in the way you intended. :)

Kind regards

robert
 
F

F. Senault

Le 19 mai à 21:29, Marc-antoine Kruzik a écrit :
Hello, I'm looking for informations about "super". /.../
I have 3 classes : grandfather, father, and son.
And they all have a method named "say_hello".
But now, I would that Son didn't call Father, but directly Grand_father.

s.say_hello
Grand Father : Hello
Son : Hello

How could I do that ?

As others said, refactor, there's a problem with you inheritance logic.

That being said, you could work in the Father class like this to obtain
the wanted result :


class Father < Grand_father
def say_hello
super
puts "Father : Hello" unless self.is_a? Son
end
end

Fred
 
R

Robert Klemme

2009/5/20 F. Senault said:
That being said, you could work in the Father class like this to obtain
the wanted result :


class Father < Grand_father
=A0def say_hello
=A0 =A0super
=A0 =A0puts "Father : Hello" unless self.is_a? Son
=A0end
end

Frankly, I find that even worse than my hack. The reason: now Father
needs to know something about a sub class so we have a circular
dependency. That's usually a bad thing to have. Consider, all these
classes are in different files and someone only needs Father and
Grand_father: code will break in Father because Son is undefined.
It's ok if Son knows something about his parents but not otherwise.

Kind regards

robert


--=20
remember.guy do |as, often| as.you_can - without end
http://blog.rubybestpractices.com/
 
K

KDr2

[Note: parts of this message were removed to make it a legal post.]

class Object
def super!(n,*args)
method=caller[0][%r/.*?`(.*?)'/,1].to_sym
klass=self.class.ancestors[n]
klass.instance_method(method).bind(self).call(*args)
end
end


class Grand_father
def say_hello
puts "Grand Father : Hello"
end
end

class Father < Grand_father
def say_hello
super
puts "Father : Hello"
end
end

class Son < Father
def say_hello
super! 2
puts "Son : Hello"
end
end

s = Son.new
s.say_hello
 
M

Marc-antoine Kruzik

Ilan said:
I am taking a break at work so I can't test it, but could you please
try:
super.super.say_hello

Too bad it doesn't work. But it was pretty logical. :)


@ KDr2

That is exactly what I needed ! :D
Thank you very much.
 
C

Codeblogger

[Note: parts of this message were removed to make it a legal post.]

2009/5/20 Marc-antoine Kruzik said:
Too bad it doesn't work. But it was pretty logical. :)
I don't think it was logical, but I guess, that's a matter of taste and
expectation of what 'super' does.
And still: KDr2's solution is bad software design.

Regards
Nicolai
 
I

Ilan Berci

Robert said:
class Son < Father
def say_hello
Grand_father.instance_method:)say_hello).bind(self).call
puts "Son : Hello"
end
end

Ahh.. cool! Thanks for the bind() solution, I am aware of it but it
always falls to the back of my head.. I blame either my age, marriage,
or kids on this one..

ilan
 
P

Pieter V.

Depending on your purpose, it may make sense to allow the caller to
specify which class to invoke the method in. =C2=A0(This won't frequently
be the case, but since you're specifically looking to invoke your
Grandfather's method, it might apply here.)

--------8<-------------------------

def Grandfather
=C2=A0=C2=A0def say_hello(klass=3Dself.class)
return super unless self.is_a?(klass)
=C2=A0=C2=A0 =C2=A0puts 'Grandfather: Hello'
=C2=A0=C2=A0end
end

def Father < Grandfather
def say_hello(klass=3Dself.class)
return super unless self.is_a?(klass)
puts 'Father: Hello'
end
end

def Son < Father
def say_hello(klass=3Dself.class)
return super(Grandfather) unless self.is_a?(klass)
puts 'Son: Hello'
end
end

--------8<-------------------------

It's not a best practice, but it might prove useful enough for your case.
 
M

Marc-antoine Kruzik

Pieter said:
Depending on your purpose, it may make sense to allow the caller to
specify which class to invoke the method in.  (This won't frequently
be the case, but since you're specifically looking to invoke your
Grandfather's method, it might apply here.)

It's not a best practice, but it might prove useful enough for your
case.

This was just an example. These methods have sometimes arguments, so I
had to find something else.

Because we found by ourselves two solutions, and yours was one of these.
The other was to use aliases, but it was bad.
 

Ask a Question

Want to reply to this thread or ask your own question?

You'll need to choose a username for the site, which only take a couple of moments. After that, you can post your question and our members will help you out.

Ask a Question

Similar Threads


Members online

No members online now.

Forum statistics

Threads
473,979
Messages
2,570,185
Members
46,728
Latest member
FernMcmull

Latest Threads

Top