Generar números de Friedman

9

Un número de Friedman es un número que se puede expresar aplicando operaciones matemáticas básicas (^, /, *, +, -) a todos sus dígitos. Las operaciones no necesitan aplicarse a cada dígito individual, pero todos los dígitos deben estar involucrados. Es decir, 121 = 11 ^ 2 -> todos los dígitos están involucrados, pero 1 y 1 se han unido para formar 11.

Se permite el uso de paréntesis, pero la solución trivial x= (x)no es una solución válida. Además, no es válida, x= +x.

Ejemplos

  • 25 = 5 ^ 2
  • 121 = 11 ^ 2
  • 343 = (3 + 4) ^ 3
  • 2048 = (8 ^ 4) / 2 + 0

Escriba un programa que tome dos enteros positivos e imprima el número de números de Friedman en ese rango (inclusive) y los números con las expresiones en las líneas siguientes.

Entrada -

n m    | n, m integers, n>=0, m>n

Salida -

count    | number of Friedman numbers in the given range
fn1 exp1 | Friedman number, expression
fn2 exp2
fn3 exp3
.
.
.

El código más corto publicado antes del domingo 29 de julio a las 00:00 Hrs GMT será el ganador.

elssar
fuente
2
¿Puedes agregar algunos ejemplos de números de Friedman y explicar cómo /funciona? Por ejemplo, ¿qué es 1/3?
JPvdMerwe
El número se expresa aplicando las operaciones a todos sus dígitos. es decir, 25 = 5 ^ 2, 126 = 6 * 21, 343 = (3 + 4) ^ 3 y así sucesivamente
elssar
¿Permiten menos unario? por ejemplo -5?
JPvdMerwe
@JPvdMer: verificamos la especificación de entrada, no es necesario que lo haga, pero si lo desea, elimínelo. Aunque unary plus no está permitido. es decir, +5 no es una solución válida
elssar
1
No ha respondido la pregunta de JPvdMerwe sobre la división. ¿Debe ser exacto? ¿Pueden los resultados intermedios ser no integrales?
Peter Taylor

Respuestas:

3

Rubí, 456 438 408 390 370 349 344 334 [fijo]

g={}
f=->a,b{a.permutation(b).to_a.uniq.flatten.each_slice b}
F,T=$*
([F.to_i,10].max..T.to_i).map{|c|f[a="#{c}".split(''),v=a.size].map{|m|f[[?+,?-,?*,?/,'','**'],v-1].map{|w|(d=(s=m.zip(w)*'').size)==v&&next
0.upto(d){|y|y.upto(d+1){|u|begin(r=eval t="#{s}".insert(y,?().insert(u,?)))==c&&g[r]=t
rescue Exception
end}}}}}
p g.size,g

Salida:

% ruby ./friedman-numbers.rb 1 300
9
{25=>"(5)**2", 121=>"(11)**2", 125=>"5**(2+1)", 126=>"(6)*21", 127=>"(2)**7-1", 128=>"2**(8-1)", 153=>"(3)*51", 216=>"6**(1+2)", 289=>"(9+8)**2"}

También funciona relativamente rápido para números más grandes:

% time ruby friedman-numbers.rb 3863 3864   
1
{3864=>"(6**4-8)*3"}
ruby friedman-numbers.rb 3863 3864  14.05s user 0.17s system 99% cpu 14.224 total
defhlt
fuente
1
Me encontré con la entrada 5 40y consiguió el resultado: [11, "11**1", 21, "21**1", 31, "31**1", 41, "41**1"]. No había señales de 25allí y creo que la solución adecuada (por ejemplo, para 21) es 2*1, no21**1
Cristian Lupascu
@ w0lf Gracias! Creo que lo he arreglado.
defhlt
Sí, funciona muy bien ahora.
Cristian Lupascu
@ w0lf agregó muchos caracteres para formatear la salida según sea necesario
defhlt
puedes ganar 2 caracteres reemplazando '+-*/'.chars.to_a+['','**']con["+","-","*","/","","**"]
Cristian Lupascu
4

Python 2.7 - 380 378 372 371 367 363 357 354 352 348 336 caracteres

Solo una simple búsqueda de fuerza bruta.

from itertools import*
s=lambda x:[x]['1'>x>'0':]+['(%s%s%s)'%f for i in range(1,len(x))for f in product(s(x[:i]),'*/-+^',s(x[i:]))]
def E(e):
 try:return eval(e.replace("^","**"))
 except:0
A={i:e for i in range(input(),input()+1)for x in permutations(`i`)for e in s("".join(x))[x>='1':]if E(e)==i}
print len(A)
for v in A:print v,A[v]

Ejemplo de ejecución:

1
300
9
128 (2^(8-1))
289 ((9+8)^2)
216 (6^(1+2))
121 (11^2)
153 (3*51)
25 (5^2)
125 (5^(2+1))
126 (6*21)
127 ((2^7)-1)

Explicación:

s(x) es una función que toma una cadena que contiene una secuencia de dígitos y devuelve todas las expresiones usando esos dígitos en ese orden.

[x]['1'>x>'0':] se evalúa como una lista que contiene x si x es '0' o una secuencia de dígitos que no comienza con '0'; de lo contrario, se evalúa como una lista vacía. Básicamente, esto maneja el caso donde unir todos los dígitos juntos.

['(%s%s%s)'%f for i in range(1,len(x))for f in product(s(x[:i]),'*/-+^',s(x[i:]))] básicamente divide x en dos partes (ambas de longitud distinta de cero), llama a s () en cada parte y une todos los resultados junto con algún operador entre ellos, usando product ().

E(e) es básicamente una evaluación segura Devuelve el valor de e si e es válido y Ninguno de lo contrario.

A={i:e for i in range(input(),input()+1)for x in permutations(`i`)for e in s("".join(x))[x>='1':]if E(e)==i}

Básicamente, este código prueba todos los números en el rango, permuta sus dígitos y prueba cada expresión que s () genera para esa permutación, ignorando la primera expresión si x no comienza con '0', porque si x no comienza con ' 0 ', entonces la primera expresión será simplemente x.

Versión alternativa - 397 caracteres

Aquí está mi código si debe usar fracciones:

from fractions import*
from itertools import*
s=lambda x:["Fraction(%s)"%x]['1'>x>'0':]+['(%s%s%s)'%f for i in range(1,len(x))for f in product(s(x[:i]),'*/-+^',s(x[i:]))]
def E(e):
 try:return eval(e.replace("^","**"))
 except:0
A={i:e for i in range(input(),input()+1)for x in permutations(`i`)for e in s("".join(x))[x>='1':]if E(e)==i}
print len(A)
for v in A:print v,A[v].replace("Fraction","")
JPvdMerwe
fuente
No creo if len(x)<2que alguna vez sea cierto en su función s. Además, puede reemplazar su formatcon "a[Fraction(%s)%s%s]='(%s%s%s)'"%(x[:i],o,v,x[:i],o,A)para guardar 4 caracteres.
beary605
@ beary605: Es cierto que a veces, cuando i = len (x) -1, la siguiente llamada obtendrá un único carácter. En cuanto al segundo punto, ¡gracias! :)
JPvdMerwe
eh ... except:0inteligente ... muy inteligente. Lo recordaré
Ev_genus
Incluya algunos resultados ilustrativos.
DavidC
1
No, todavía corriendo. Tengo que mover mi PC ahora, pero dejaré que se ejecute durante unos días y veré si termina.
JPvdMerwe
3

Python3 (436) (434) (443)

Fue dificil. Puedo ahorrar algunos caracteres si hago que la salida sea más nativa.

from itertools import*
r={};k=product;m=map
q=lambda n,h=1:["("+i+c+j+")"for(i,j),c in k(chain(*[k(*m(q,f))for f in sum(([(x[:q],x[q:])for q in range(1,len(x))]for x in m("".join,permutations(n))),[])]),list("+-*/^")+[""]*h)]if 1<len(n)else[n]*h
a,b=m(int,m(input,"nm"))
for i,j in chain(*[k(q(str(n),0),[n])for n in range(a,b+1)]):
    try:exec("if eval(%r)==j:r[j]=i"%i.replace("^","**"))
    except:0
print(len(r))
for j,i in r.items():print(i,j)

Salida

n100
m200
6
(2^(8-1)) 128
(3*(51)) 153
((11)^2) 121
(5^(1+2)) 125
(6*(21)) 126
((2^7)-1) 127
Ev_genus
fuente
1
Entonces tienes muchos trucos ingeniosos; Sin embargo, debo mencionar que no manejas del 1 al 9 correctamente y tu entrada no es inclusiva. Se puede quitar 2 caracteres aunque quitando el espacio después "("+i+c+j+")"y su sustitución len(n)>1por 1<len(n)después de lo cual se puede quitar el espacio después de esa expresión.
JPvdMerwe
Justa. Se
corrigieron
Puede reemplazar la última línea for j in r:print(r[j],j)para guardar 7 caracteres.
JPvdMerwe
1

Mathematica 456 416 402 404 400 396 caracteres

<< Combinatorica`; l = Length; p = Permutations; f = Flatten; c = Cases;
u[d_, o_, s_] := 
 Fold[#2[[1]] @@ If[s == 1, {#1, #2[[-1]]}, {#2[[-1]], #1}] &, 
 d[[1]], Thread@{o, Rest@d}];
q[t_, r_] := {u[t, #, r], u[HoldForm /@ t, #, r]} & /@ 
p[{Plus, Subtract, Times, Divide, Power}, {l@t - 1}];
v[m_, n_] := (t = Table[Union@
  c[f[{#~q~1, #~q~0} & /@ 
     f[p /@ c[
        FromDigits /@ # & /@ 
         f[SetPartitions /@ p@IntegerDigits@j, 1], x_ /; l@x > 1],
       1], 2], {j, _}], {j, m, n}]~f~1; {l@t}~Join~t)

Ejemplo :

v[1,300]//TableForm

Salida :

salida de Friedman

DavidC
fuente