Precisão arbitrária de ponto flutuante Python disponível?

Só por diversão e porque foi muito fácil, eu escrevi um programa curto para gerarEnxertando números, mas por causa de problemas de precisão de ponto flutuante não está encontrando alguns dos exemplos maiores.

def isGrafting(a):
  for i in xrange(1, int(ceil(log10(a))) + 2):
    if a == floor((sqrt(a) * 10**(i-1)) % 10**int(ceil(log10(a)))):
      return 1

a = 0
while(1):
  if (isGrafting(a)):
    print "%d %.15f" % (a, sqrt(a))
  a += 1

Este código perde pelo menos um número de enxerto conhecido.9999999998 => 99999.99998999999999949999999994999999999374999999912... Parece cair precisão extra depois de multiplicar por10**5.

>>> a = 9999999998
>>> sqrt(a)
99999.99999
>>> a == floor((sqrt(a) * 10**(5)) % 10**int(ceil(log10(a))))
False
>>> floor((sqrt(a) * 10**(5)) % 10**int(ceil(log10(a))))
9999999999.0
>>> print "%.15f" % sqrt(a)
99999.999989999996615
>>> print "%.15f" % (sqrt(a) * 10**5)
9999999999.000000000000000

Então eu escrevi um pequeno programa em C ++ para ver se era minha CPU truncando o número de ponto flutuante ou python de alguma forma.

#include <cstdio>
#include <cmath>
#include <stdint.h>

int main()
{
  uint64_t a = 9999999998;
  printf("%ld %.15f %.15f %.15f %.15f\n", a, sqrt((double)a), sqrt((double)a)*1e4, sqrt((double)a)*1e5, sqrt((double)a)*1e6);
  a = 999999999998;
  printf("%ld %.15f %.15f %.15f %.15f\n", a, sqrt((double)a), sqrt((double)a)*1e5, sqrt((double)a)*1e6, sqrt((double)a)*1e7);
  a = 99999999999998;
  printf("%ld %.15f %.15f %.15f %.15f\n", a, sqrt((double)a), sqrt((double)a)*1e6, sqrt((double)a)*1e7, sqrt((double)a)*1e8);
  return 0;
}

Quais saídas:

9999999998 99999.999989999996615 999999999.899999976158142 9999999999.000000000000000 99999999990.000000000000000
999999999998 999999.999998999992386 99999999999.899993896484375 999999999999.000000000000000 9999999999990.000000000000000
99999999999998 9999999.999999899417162 9999999999999.900390625000000 99999999999999.000000000000000 999999999999990.000000000000000

Então parece que estou correndo muito contra os limites da precisão de ponto flutuante e a CPU está cortando os bits restantes porque acha que a diferença restante é erro de ponto flutuante. Existe uma maneira de contornar isso em Python? Ou eu preciso mudar para C e usar GMP ou algo assim?

questionAnswers(5)

yourAnswerToTheQuestion