views:

354

answers:

3

So I understand you aren't supposed to to directly subclass Fixnum, Float or Integer, as they don't have a #new method. Using DelegateClass seems to work though, but is it the best way? Anyone know what the reason behind these classes not having #new is?

I need a class which behaves like a Fixnum, but has some extra methods, and I'd like to be able to refer to its value through self from within the class, for example:

class Foo < Fixnum
  def initialize value
    super value
  end

  def increment
    self + 1
  end
end

Foo.new(5).increment + 4 # => 10
+2  A: 

Could you not extend FixNum itself? Like...

class Fixnum
  def even?
    self % 2 == 0
  end
end

42.even?
Turnor
I could, but I'd rather not. Also, Fixnum doesn't have #new, and I need to be able to add functionality to #initialize.
cloudhead
+3  A: 

IIRC, the main implementation of Ruby stores Fixnums as immediate values, using some of the low bits of the word to tag it as a Fixnum instead of a pointer to an object on the heap. That's why, on a 32-bit machine, Fixnums are only 29-bits (or whatever it is) instead of a full word.

So because of that, you can't add methods to a single "instance" of Fixnum, and you can't subclass it.

If you're dead-set on having a "Fixnum-like" class, you'll probably have to make a class that has a Fixnum instance variable, and forward method calls appropriately.

Jason Creighton
ah I see. Is there any other class I could subclass? Numeric has a #new, but it doesn't take any arguments, so I suppose it's an abstract class.
cloudhead
Hmm. I haven't touched Ruby in a while, and I don't recall exactly how the numeric class tree is laid out.
Jason Creighton
The only obvious candidate is BigDecimal.
Yehuda Katz
+5  A: 

You can pretty easily set up a quick forwarding implementation yourself:

class MyNum
  def initialize(number)
    @number = number
  end

  def method_missing(name, *args, &blk)
    ret = @number.send(name, *args, &blk)
    ret.is_a?(Numeric) ? MyNum.new(ret) : ret
  end
end

Then you can add whatever methods you want on MyNum, but you'll need to operate on @number in those methods, rather than being able to call super directly.

Yehuda Katz
Hmm. But if you did MyNum.new(2) + MyNum.new(3), wouldn't you get a Fixnum back instead of a MyNum instance? I think you need to wrap @number.send in MyNum.new
Jason Creighton
I updated my answer to ensure that you return a MyNum if the original response was a Numeric.
Yehuda Katz