quine-ish kółko i krzyżyk

19

Napisz program w wybranym języku, który gra idealną grę w kółko i krzyżyk na planszy 3 * 3 przeciwko ludzkiemu graczowi. Jednak każdy ruch musi być innym programem , wygenerowanym z poprzedniej iteracji.

To, jak iw jakiej formie oceniasz wkład człowieka, zależy od ciebie, ale należy go odczytać ze standardowego wkładu. Podobnie, masz swobodę wyboru metody, w jaki sposób ustalisz, który gracz zacznie (na przykład najpierw pytasz lub pozwalasz człowiekowi wprowadzić nieprawidłowy ruch, aby zasygnalizować uruchomienie komputera lub inne pomysły).

Sprawdzanie poprawności ruchów nie jest konieczne, możesz założyć uczciwego przeciwnika.

Zasadniczo masz program, który odpowiada stanowi płyty. Stan jest drukowany w dowolny rozpoznawalny sposób, ale oczekuje się co najmniej następującego poziomu szczegółowości:

X..
00X
x..

Po tym, jak ludzki gracz wprowadzi swoje ruchy, twój program musi wygenerować następną iterację jako plik źródłowy w tym samym języku (na standardowe wyjście lub do pliku) i zakończyć. Nie wolno przechowywać informacji w żadnym innym miejscu poza tym plikiem źródłowym. (nie jest konieczne, aby Twój program skompilował i uruchomił wygenerowany program, może to zrobić użytkownik - jednak nie jest to zabronione). Po wygenerowaniu i uruchomieniu wygenerowanego programu będzie on zachowywać się podobnie, wyświetlać stan, czekać na dane wejściowe użytkownika itp.

Pod koniec gry musisz wydrukować wynik (niezależnie od tego, czy wygrałeś, czy remis) w jakikolwiek jednoznaczny sposób.

Przez idealną grę mam na myśli, że program nie może przegrać, a jeśli istnieje możliwość wymuszenia wygranej, powinien wygrać.

Wygrywa najkrótszy kod , zwycięzca jest wybierany co najmniej 10 dni po pierwszym ważnym wpisie.

Otrzymasz 10% obniżkę wyniku, jeśli twój program poradzi sobie z budową i uruchomieniem kolejnej iteracji. (Wiem, najprawdopodobniej nie warto) Oczywiście sam program musi zostać zakończony do czasu, gdy następna iteracja zaakceptuje ruchy użytkownika.

Jeśli używasz dziwnych, nietypowych sztuczek, opublikuj krótkie wyjaśnienie ze swoim kodem.

vsz
źródło
2
Niezłe wyzwanie, ale myślę, że zrezygnuję.
John Dvorak
„Każdy ruch musi być innym programem”. Czy masz na myśli: „każda gra musi być inicjowana i zarządzana przez nową, odrębną instancję oryginalnego programu”?
DavidC
1
@DavidCarraher: Nie. Każdy ruch, nie tylko każda gra. Sprawdź opis poniżej przykładu tablicy. Kiedy komputer musi wykonać ruch (więc zmienia się stan płyty), twój program musi wygenerować plik źródłowy, który po zbudowaniu i uruchomieniu stanie się następnym stanem. Oryginalny program zostanie zamknięty. Nowo wygenerowany program, wykonując ruch, będzie zachowywał się podobnie: tworzy plik źródłowy, który po zbudowaniu i uruchomieniu stanie się kolejnym stanem i tak dalej. Ponieważ nie jest dozwolone przechowywanie informacji, z wyjątkiem wygenerowanego pliku źródłowego, przypomina to quine z różnicami między iteracjami.
vsz

Odpowiedzi:

13

Perl, 933 znaków

$m=<<'';$_='         ';
sub h{/^(?:...)*(\d)\1\1/|/^.?.?(\d)..\1..\1/|/(\d)...\1...\1/|/^..(\d).\1.\1/&&$1}
sub r{substr($_,$p-1,1)=pop}sub p{my$w=pop;my@b=(0,h==$w||h&&-1);if(!$b[1]&&/ /){$b[1]=-9;
while(/ /g){local($_,$p)=($_,pos);r$w;$r=-(p($w^1))[1];@b=($p,$r)if$r>$b[1]}}@b}
if(($w=h||!/ /)||!@ARGV){$w--&&print+(nobody,X,O)[$w]," wins\n";s/(...)/$1\n/g;
tr/ 23/.XO/;print}else{$w=3;$w^=1for/\d/g;($p=pop)?r($w^1)&&!h&&(($p)=p$w)&&r$w:s/ /2/;
print"\$m=<<'';\$_='$_';\n$m\n$m"}

sub h{/^(?:...)*(\d)\1\1/|/^.?.?(\d)..\1..\1/|/(\d)...\1...\1/|/^..(\d).\1.\1/&&$1}
sub r{substr($_,$p-1,1)=pop}sub p{my$w=pop;my@b=(0,h==$w||h&&-1);if(!$b[1]&&/ /){$b[1]=-9;
while(/ /g){local($_,$p)=($_,pos);r$w;$r=-(p($w^1))[1];@b=($p,$r)if$r>$b[1]}}@b}
if(($w=h||!/ /)||!@ARGV){$w--&&print+(nobody,X,O)[$w]," wins\n";s/(...)/$1\n/g;
tr/ 23/.XO/;print}else{$w=3;$w^=1for/\d/g;($p=pop)?r($w^1)&&!h&&(($p)=p$w)&&r$w:s/ /2/;
print"\$m=<<'';\$_='$_';\n$m\n$m"}

Należy pamiętać, że pusta linia pośrodku skryptu faktycznie musi tam być. (Podziały linii na końcu długich linii nie są wymagane, z wyjątkiem czytelności, i nie są uwzględniane w liczbie znaków.)

Zastosowanie: Gdy program jest uruchamiany bez argumentów, wyświetla bieżący stan gry. Ponieważ na początku tablica jest pusta, wyjście będzie:

...
...
...

Uruchom program z argumentem od 1 do 9, aby przejąć ten kwadrat jako ruch. Program wykona własny ruch, a następnie wyśle ​​skrypt zastępczy z nowym stanem. Zatem na przykład:

$ perl ./qttt 5 > ./qttt-2
$ perl ./qttt-2
O..
.X.
...

Tylko w pierwszej turze możesz wykonać ruch, 0aby wskazać, że komputer powinien wykonać pierwszy ruch. Pamiętaj, że zawsze pierwszym graczem będzie X.

Po zakończeniu gry na wyświetlaczu pojawi się informacja na ten temat:

$ perl ./qttt-4 6 > ./qttt-5
$ perl ./qttt-5
O wins
OXX
OOX
X.O

Program działa poprzez standardowe wyszukiwanie drzewa gry za pomocą minimax. (Kółko i krzyżyk to gra na tyle mała, że ​​przy każdym uruchomieniu można wygenerować pełne drzewo gry). Wyjątkiem jest sytuacja, w której komputer porusza się pierwszy - w tym przypadku początkowe przejście do lewego górnego rogu jest trudne zakodowane.

Zauważ, że ten program działa w sposób zapewniający odpowiednią quine - w żadnym momencie skrypt nie uzyskuje dostępu do własnego pliku źródłowego w celu wygenerowania danych wyjściowych.

chlebak
źródło
1
To jest piękne! Nie zdawałem sobie sprawy, że przez najdłuższy czas gapiłem się na ogromną dokumentację, a potem zrobiłem to podwójnie.
Jesse Smith