Arbitrary precision arithmetic with Ruby

前端 未结 5 804
醉话见心
醉话见心 2021-01-13 04:18

How the heck does Ruby do this? Does Jörg or anyone else know what\'s happening behind the scenes?

Unfortunately I don\'t know C very well so bignum.c is of little

相关标签:
5条回答
  • 2021-01-13 04:27

    I don't know of the implementation details so I'll cover how a basic Big Number implementation would work.

    Basically instead of relying on CPU "integers" it will create it's own using multiple CPU integers. To store arbritrary precision, well lets say you have 2 bits. So the current integer is 11. You want to add one. In normal CPU integers, this would roll over to 00

    But, for big number, instead of rolling over and keeping a "fixed" integer width, it would allocate another bit and simulate an addition so that the number becomes the correct 100.

    Try looking up how binary math can be done on paper. It's very simple and is trivial to convert to an algorithm.

    0 讨论(0)
  • 2021-01-13 04:28

    Simple: it does it the same way you do, ever since first grade. Except it doesn't compute in base 10, it computes in base 4 billion (and change).

    Think about it: with our number system, we can only represent numbers from 0 to 9. So, how can we compute 6+7 without overflowing? Easy: we do actually overflow! We cannot represent the result of 6+7 as a number between 0 and 9, but we can overflow to the next place and represent it as two numbers between 0 and 9: 3×100 + 1×101. If you want to add two numbers, you add them digit-wise from the right and overflow ("carry") to the left. If you want to multiply two numbers, you have to multiply every digit of one number individually with the other number, then add up the intermediate results.

    BigNum arithmetic (this is what this kind of arithmetic where the numbers are bigger than the native machine numbers is usually called) works basically the same way. Except that the base is not 10, and its not 2, either – it's the size of a native machine integer. So, on a 32 bit machine, it would be base 232 or 4 294 967 296.

    Specifically, in Ruby Integer is actually an abstract class that is never instianted. Instead, it has two subclasses, Fixnum and Bignum, and numbers automagically migrate between them, depending on their size. In MRI and YARV, Fixnum can hold a 31 or 63 bit signed integer (one bit is used for tagging) depending on the native word size of the machine. In JRuby, a Fixnum can hold a full 64 bit signed integer, even on an 32 bit machine.

    The simplest operation is adding two numbers. And if you look at the implementation of + or rather bigadd_core in YARV's bignum.c, it's not too bad to follow. I can't read C either, but you can cleary see how it loops over the individual digits.

    0 讨论(0)
  • 2021-01-13 04:29

    You could read the source for bignum.c...

    At a very high level, without going into any implementation details, bignums are calculated "by hand" like you used to do in grade school. Now, there are certainly many optimizations that can be applied, but that's the gist of it.

    0 讨论(0)
  • 2021-01-13 04:33

    It uses the Bignum class

    irb(main):001:0> (999**999).class
    => Bignum
    

    Rdoc is available of course

    0 讨论(0)
  • 2021-01-13 04:34

    Beaconaut APICalc 2 just released on Jan.18, 2011, which is an arbitrary-precision integer calculator for bignum arithmetic, cryptography analysis and number theory research......

    http://www.beaconaut.com/forums/default.aspx?g=posts&t=13

    0 讨论(0)
提交回复
热议问题