Recupere el código fuente mutado

27

En un accidente muy inusual que involucra una pequeña muestra de radio, una ballena electrocutada y tres ositos de goma, se ha mutado parte del código fuente de The Management ™. Poco sabe el jefe de The Management ™, en realidad fueron los policías los responsables, en un intento por frustrar los planes "malvados" de The Management ™. Entonces, los Robbers® han sido contratados en un intento de recuperar el código original, porque ¿a quién no le gusta ser malvado a veces?

nota: Este desafío se inspiró en gran medida en Descifrar el código fuente .

Descripción

Este es un desafío de .

  • Los policías escribirán un programa (el código mutado) que realiza la Tarea # 1 (y también escriben un programa que realiza la Tarea # 2, pero se mantiene en secreto).
  • Los ladrones intentarán revertir la "mutación" y cambiar este código original en un código que realice la Tarea # 2.

En este desafío, la Tarea n. ° 1 será la salida del nnúmero primo y la Tarea n. ° 2 será el nnúmero de Fibonacci (que de alguna manera es malo, de acuerdo con los Cops © de todos modos). La secuencia de Fibonacci se define como ( n=11; n=21; n=32; ...), y los números primos se definen como ( n=12; n=23; n=35; ...).

El objetivo de los policías es minimizar la diferencia entre los programas que completan la Tarea n. ° 1 y la Tarea n. ° 2, mientras se evita que los ladrones vuelvan a crear el código que completa la Tarea n. ° 2.

Reglas de policía

Los policías escribirán dos programas (uno que complete la Tarea # 1 y otro que complete la Tarea # 2), y harán pública la siguiente información:

  • El primer programa (que genera el nnúmero primo th)
  • La distancia de edición de Levenshtein entre el primer programa y el segundo programa
  • El lenguaje de programación en el que ambos programas están escritos (debe ser el mismo lenguaje para ambos programas)

Las siguientes restricciones se aplican a ambos programas:

  • Deben tener 128 caracteres de longitud o menos.
  • Solo deben usar ASCII imprimible (más líneas nuevas, que también están permitidas).
  • Deben tardar menos de 10 segundos en ejecutarse n=45y no están obligados a producir la salida correcta para ninguno n>45.
  • No deben usar ninguna función hash o criptográfica.

Reglas de ladrones

El ladrón intentará cambiar el programa del policía (que completa la Tarea # 1) en un programa que complete la Tarea # 2 (no necesariamente el programa original escrito por el policía) en la distancia de edición especificada por el policía.

Un envío ya agrietado no se puede volver a agrietar (solo el primer ladrón que descifra un envío obtiene crédito).

Después de descifrar un envío, haga lo siguiente:

  • Publique una respuesta a la pregunta adjunta (enlace) de este desafío , proporcionando el idioma, su solución y un enlace a la respuesta original.
  • Deje un comentario con el texto "Agrietado" que se vincula a su respuesta publicada.
  • Edite la respuesta del policía si tiene privilegios de edición (si no los tiene, espere hasta que otra persona con los privilegios necesarios lo haga por usted o sugiera una edición).

Tanteo

Si el programa de un policía permanece sin descifrar durante 1 semana, el policía puede publicar el código original que completa la Tarea # 2 (en la distancia de edición especificada), y el envío se considera a partir de ese momento "seguro". La presentación segura que tenga la menor distancia de edición ganará. En caso de empate, gana el programa más corto (el original que completa la Tarea # 1). Si dos envíos aún están empatados, el que se publicó anteriormente gana.

Si un ladrón descifra con éxito el envío de un policía, la puntuación del ladrón aumenta en la distancia de edición de ese envío. Por ejemplo, un ladrón que descifra un envío con una distancia de edición de 3 y uno con una distancia de 5 gana 8 puntos. El ladrón con el puntaje más alto gana. En caso de empate, el ladrón que obtuvo el puntaje primero gana.

Tabla de clasificación

  1. Ruby, 6 (histocrat)

Una pequeña herramienta para calcular la distancia de Levenshtein

Pomo de la puerta
fuente
1
¿Cuál es el primer número de Fibonacci? 0 o 1? O no importa
kukac67
@ kukac67 Es 1; He editado la publicación.
Pomo de la puerta
¿Cuál debería ser la salida de los programas, en caso de desbordamiento?
es1024
¿Debe ser un programa completo o puede ser una función? ¿Qué pasa con una función anónima?
Tyilo
2
¿Qué se considera una "función hash o criptográfica"? ¿Puedo convertir cosas de base? ¿Puedo tomar exponenciales grandes módulos primos grandes?
Martin Ender

Respuestas:

6

Python 2, distancia = 8 [ agrietada ]

from fractions import*
n=input()
k,=P=[2]
while n>len(P):k+=1;z=reduce(lambda x,y:x*y,P,1);P+=[k]*(gcd(z,k)<2)
print P[-1]

Finalmente conseguí este debajo del límite de char. No debería ser demasiado difícil, pero pensé que la idea era interesante.


Solución prevista:

from fractions import* n=input() k,=P=[1] while n>len(P):k+=1;z=reduce(lambda x,y:x+y,P[:-1],1);P+=[z]*(gcd(z,k)<2) print P[-1]

La idea era usar eso F(n+2) = 1 + (sum over F(k) from k = 1 to n), y el hecho de que los números consecutivos de Fibonacci son coprimos. Se 1suponía que el argumento in the reduce proporcionaba el +1.

¡Parece que Feersum encontró una línea de ataque diferente!

Sp3000
fuente
Agrietado
feersum
4

Ruby, distancia 6 [seguro]

require'prime';p Prime.take(n=gets.to_i)[-1]
#p (((807462154311276410)**n/(5**0.5)).round)

Crear pares de fórmulas con distancias cortas de edición es divertido, pero parece que este enfoque podría ser más efectivo / molesto. Puede entender exactamente lo que hice, pero eso no significa que pueda revertirlo.

Solución:

require'prime';p=Prime.take(n=gets.to_i)[-1] p (((0742154311276/4e10)**n/(5**0.5)).round)

Explicación:

El código genera la proporción áurea a 11 decimales y la usa para calcular directamente la secuencia de Fibbonaci. Es suficiente precisión para obtener el número requerido de términos correctos. Esa parte no se ofuscó en absoluto, si conoces la fórmula. Para dificultar que la fuerza bruta revierta mis mutaciones y recupere la constante, utilicé la notación octal (el 0 principal) y la notación científica (4e10). Dividir entre 4e10 en lugar de 1e11 hace que parezca que estoy dividiendo entre algo .0para forzar la división de flotación, cuando en realidad cualquier cosa en notación científica es, por alguna razón, siempre una Flotación en Rubí, incluso cuando un Bignum parece tener más sentido. Pensé que estaba siendo inteligente con las p=cosas, pero la forma en que las escribí puede eliminarlas p. Yo podría'p=solución usando en p&&lugar de #en la segunda línea, pero no pensé en ello.

histocrat
fuente
No pensé en intentar insertar una eahí abajo cuando hacía fuerza bruta. Solución realmente disimulada. :)
Vectorizado el
3

Python 2 - LD = 13 agrietado

n=1;j=input();
while j>0:
    n+=1;j-=1;
    while~-all(n%i for i in range(2,n)):n+=1;
print n

Una agradable, fácil (espero que no sea demasiado fácil) para comenzar las cosas :)

Parece que fue demasiado fácil;) Me siento bastante tonto porque olvidé que podrías usar comentarios: /

FryAmTheEggman
fuente
Agrietado.
Sp3000
3

Haskell, distancia = 13

import Data.List
x=id=<<snd(mapAccumL(\m n->(,)=<<(++m)$[n|and[n`mod`m1/=0|m1<-m]])[1+1][3..])
main=print.(!!)(0:2:x)=<<readLn

Esto podría ser más legible, pero se importcomió demasiados bytes, así que tuve que jugar un poco.

Zgarb
fuente
2

Rubí, distancia 14 ( Agrietada )

p [x=2,y=1,*(1..200).map{|i|y==(y*x**(i-1)+x%2).divmod(i)[x-1]?i:1}].-([x-1])[gets.to_i-1]
histocrat
fuente
¿Agrietado?
Vectorizado
Hm, su secuencia de Fibbonaci comienza con 0, donde las reglas dicen que comience con 1. De lo contrario, se verifica (aunque es muy diferente de mi solución prevista).
histocrat
Ok, arreglado. Buen uso de por cierto Fermat.
Vectorizado
2

CJam, distancia 10 (agrietada)

1l~{{)_mp!}g}*

Solo ponte nSTDIN. Pruébalo aquí.

Como referencia, la solución original utilizaba lo raro j.

Original:

T1]l~\{(_j\(j+}j

Martin Ender
fuente
1
Agrietado
Optimizador
2

J, distancia = 4 [seguro]

f =: 3 :  '{. (p:) (+) / 0 , y - 1x'

Solución:

f =: 3: '{. 2 (x :) (+%) / 0, y $ 1x '

Método:

Denominador {. 2(x:)de la fracción continua (+%)1 + 1 / (... (1 + 1 / (1 + 1 / (1 + 1 / (1))))).

randomra
fuente
1

Python 3, distancia = 14 [ agrietada ]

n = int(input())
P = [2]
k = 2
while n > len(P):
 k += 1
 for x in P:
  if k%x == 0: break
 else: P += [k]
print(k)

Tenía algunos caracteres adicionales, así que puse un poco de espacio en blanco para mayor claridad :)

Sp3000
fuente
Agrietado
feersum
1

JAGL Alpha 1.2 - Distancia = 16 [ Agrietado ]

T~2S]{]S1{D[dmn}wDS}wSP

No debería ser demasiado difícil, veremos qué pasa ...

globby
fuente
Agrietado ?
matsjoyce
1

TI-BASIC , distancia 38

Input N:{2>L1:For(A,3,E99:If min(1=gcd(A,seq(A,A,2,$(A:A>L1(1+dim(L1:End:L1(N

>representa la STO→clave y $representa el símbolo de la raíz cuadrada.

Timtech
fuente
44
Algunos de estos personajes no parecen ser ASCII imprimibles.
Feersum
@feersum Gracias, corregido. La distancia sigue siendo 38.
Timtech
1

Python 2 - distancia = 12 [ Agrietado ]

Estoy un poco contento con cómo resultó esto.

f=lambda p,i:p if p[45:]else f(p+[i]if all(i%q for q in p[1:])else p,-~i)
print f([1,2,3],2)[input()]

Veamos cuánto tiempo lleva ... supongo que todavía estará resquebrajado.

Editar: código acortado un poco, sin efecto en la operación / distancia.

Solución prevista

Traté de no hacer comentarios o cambios en la nueva línea.

f=lambda p,i:p if p[45:]else f(p+[p[i]+p[-2]]if all(i|q for q in p[1:])else p,-~i) print f([1,1,1],2)[input()]

PurkkaKoodari
fuente
Agrietado
feersum
0

Python 3 - Distancia = 14 [ Agrietado ]


a,c,n=1,2,int(input())
while n-1:
 c+=1
 while 1!=list(map(c.__mod__,range(2,46))).count(0):
  c,a=a+c,a
 n-=1
print(c)

Veremos cuánto dura esto ...

matsjoyce
fuente
Agrietado
feersum