Implementar un autómata extraño

11

Estaba jugando con un autómata celular y encontré uno que tenía un comportamiento interesante. Así es como funciona:

Lee una cadena binaria de izquierda a derecha, si encuentra un 1seguido por 2otros valores, agregará un 0al resultado y continuará leyendo. Si encuentra a 0(o quedan menos de 3 valores) agregará el valor actual y a 1y continuará leyendo. Al final de la cadena, agregará un solo 1al resultado.

Aquí hay un ejemplo resuelto de una generación

01011111
^

Primero nos encontramos con un, 0así que agregamos 01nuestro resultado

01011111
 ^
01

Ahora nos encontramos con un, 1así que agregamos un cero y omitimos los siguientes dos valores

01011111
    ^
010

Nos encontramos con otro, 1así que hacemos lo mismo.

01011111
       ^
0100

Ahora tenemos otro 1espacio pero no suficiente para saltar, así que agregamos la celda actual y un 1(en este caso 11)

01011111
        ^
010011

Estamos al final, así que agregamos un single 1y terminamos esta generación

01011111
        ^
0100111

Tarea

Dada la entrada en cualquier formato razonable, debe crear una función o programa que calcule una generación del autómata.

Esta es una pregunta de , por lo que las respuestas se puntuarán en bytes, con menos bytes mejor.

Implementación de muestra

Aquí hay una implementación de muestra en Haskell (define una función d, pero el programa imprime una iteración indefinidamente):

d('1':_:_:x) = "0" ++ d x
d(a:x) = a:'1':d x
d x = "1"
r x = x:map d(r x)

Pruébalo en línea!

Ad Hoc Garf Hunter
fuente
En su pregunta, usted indica Ahora tenemos otro 1 pero no suficiente espacio para saltar, por lo que agregamos la celda actual y un 1 u 11 . ¿Es 1 u 11?
caird coinheringaahing
2
Entonces, si tenemos un 10, ¿debería imprimirse 11011? Creo que unos cuantos casos de prueba sería de gran ayuda
nmjcman101
2
@WheatWizard Agradecería una explicación más clara, tal vez una tabla, de las reglas
Alexander - Restablece a Monica el
2
No creo que esto sea realmente un autómata celular, pero siéntete libre de iluminarme con una definición que lo indique.
feersum
2
@feersum De hecho, no conserva el número de células. Es un transductor de estado finito .
Ørjan Johansen

Respuestas:

5

V , 26 22 21 bytes

¡Gracias a @CowsQuack por 4 bytes combinando expresiones regulares! Y @ ØrjanJohansen para otro byte con algunas combinaciones de expresiones regulares.

Ó1../3
Ó./&1
Ó31/0
A1

Pruébalo en línea!

Utiliza sustituto varias veces y agrega un 1 al final. Nada muy elegante. Tengo una versión que se reasigna 1y 0en modo de inserción para obtener el efecto deseado, pero es bastante más largo.

(Versión de reemplazo múltiple: ¡ Pruébelo en línea! )

nmjcman101
fuente
La segunda y tercera expresiones regulares pueden fusionarse en Ó1ü0/&1( ües \|)
user41805
¡Genio de @Cowsquack!
nmjcman101
Es aún más corto de hacer Ó./&1seguido de Ó31/0.
Ørjan Johansen
3

JavaScript (ES6), 56 bytes

Toma entrada como una matriz de caracteres. Devuelve una cadena o el número 1si se le da una matriz vacía.

f=([v,...a])=>v?(+v&&a[1]?a.splice(0,2)&&'0':v+1)+f(a):1

Manifestación

Versión animada

Ejemplos de entradas estables: 0101, 010011111

Arnauld
fuente
2

Python 2 , 89 bytes

x=input()
y=0
k=[]
while x[y:]:v=1-x[y]*(y<len(x)-2);k+=[x[y]]*v+[v];y+=3-2*v
print k+[1]

Pruébalo en línea!

-4 bytes gracias a Rod
-6 bytes gracias a ovs
-1 byte gracias a micsthepick

Hiperneutrino
fuente
[0]if v else[x[y],1]puede reescribirse como [[x[y],1],[0]][v], pero puede invertir el vvalor para llegar a 96 bytes
Rod
90 bytes
ovs
Los paréntesis no son necesarios para la declaración de impresión en python 2, por lo que puede guardar un byte
micsthepick
2

Swift 3 , 147 bytes

-1 gracias a @ Mr.Xcoder

func g(i:[Int]){var r=[Int]();var s=ArraySlice(i);while let e=s.popFirst(){if 0<e&&2<s.count{r+=[0];s=s.dropFirst(2)}else{r+=[e,1]}};print(r+[1])}

Sin golf, devolviendo el valor en lugar de imprimir:

func iterate(state: [Int]) -> [Int] {
    var result = [Int]()

    var inputSlice = ArraySlice(state)

    while let element = inputSlice.popFirst() {
        if 0 < element && 2 < inputSlice.count { 
            result += [0]
            inputSlice = inputSlice.dropFirst(2)
        }
        else {
            result += [element, 1]
        }

        //debugPrint(result.map(String.init).joined(separator: ""))
    }

    return result + [1]
}
Alexander - Restablece a Monica
fuente
1
Se puede reemplazar 3<=s.countcon el 2<s.countde -1 bytes .
Sr. Xcoder
@ Mr.Xcoder Gracias! También puedo detectar 1s en la entrada en 0 < elementlugar deelement == 0
Alexander - Restablecer Monica
1

Python 2 , 81 bytes

Tanto la entrada como la salida son listas (gracias a Erik the Outgolfer)

def f(Z):return Z and((1>Z[0]or 3>len(Z))and[Z[0],1]+f(Z[1:])or[0]+f(Z[3:]))or[1]

Pruébalo en línea!

Algunos casos

[0,1,0,1,1,1,1,1] --> [0,1,0,0,1,1,1]
[0] ----------------> [0,1,1]
[1] ----------------> [1,1,1]
[] -----------------> [1]
[0,1] --------------> [0,1,1,1,1]
[1,0] --------------> [1,1,0,1,1]

Python 2 , 85 bytes

Tanto la entrada como la salida son cadenas (solución inicial)

def f(Z):return Z and(('0'==Z[0]or 3>len(Z))and Z[0]+'1'+f(Z[1:])or'0'+f(Z[3:]))or'1'

Pruébalo en línea!

Algunos casos

'01011111'--> 0100111
'0'---------> 011
'1'---------> 111
''----------> 1
'01'--------> 01111
'10'--------> 11011

Explicación Es simplemente un golf de un método recursivo.

mdahmoune
fuente
Usar listas es más corto.
Erik the Outgolfer
@EriktheOutgolfer gracias :)
mdahmoune
Ah, y puedes hacerlo en 1>Z[0]lugar de 0==Z[0].
Erik the Outgolfer
0

Scala , 131 + 29 = 160 bytes

Esto está dentro de una función que toma la cadena acomo parámetro y devuelve la salida como una cadena.

var s=""
var k=0
for(c<-0 to a.length-1)breakable{if(k>0){k-=1
break}
if(a(c)==49&c<a.length-3){s+="0"
k+=2}else s+=a(c)+"1"}
s+"1"

Tengo que hacerlo import util.control.Breaks._, así que necesito agregar esos 28 bytes más un avance de línea final.

Pruébalo en línea!

V. Courtois
fuente
0

C # (.NET Core) , 108 bytes

n=>{var t="";for(int i=0,b=n.Length;i<b;){if(n[i]>'0'&i+2<b){t+="0";i+=3;}else t+=n[i++]+"1";}return t+"1";}

Pruébalo en línea!

La entrada se toma como una cadena y una cadena se devuelve como salida.

jkelm
fuente