Catur - Temukan semua langkah hukum (kecuali castling dan en passant)


19

Tulis kode terpendek yang menghitung semua kemungkinan pergerakan (saat ini) dari pemain saat ini dari string FEN yang diberikan. Apa itu string FEN? (Wikipedia)

  • Kode terpendek menang, bahasa tidak masalah.
  • Keluaran yang bergerak harus mematuhi aturan pergerakan catur kecuali yang lewat , castling, dan promosi pion.
  • Abaikan memeriksa, skakmat, dan jalan buntu, raja tidak bisa ditangkap situasi juga.

Anda dapat mengatur output berbeda seperti yang Anda inginkan (misalnya: A2-A4, A2A4,a2a4 , a2->a4...)

Kasus uji:

# INPUT 1:

rnbqkbnr / pppppppp / 8/8/8/8 / PPPPPPPP / RNBQKBNR dengan KQkq - 0 1
# OUTPUT 1

A2-A4, A2-A3, B2-B4, B2-B3, C2-C4, C2-C3, D2-D4, D2-D3, E2-E4, E2-E3,
F2-F4, F2-F3, G2-G4, G2-G3, H2-H4, H2-H3, B1-A3, B1-C3, G1-F3, G1-H3
# INPUT 2

7k / 8/8/8/8/8 / PP6 / Q1q4K w - - 0 1

# OUTPUT 2

A1-B1, A1-C1, A2-A3, A2-A4, B2-B3, B2-B4, H1-H2, H1-G1, H1-G2

juga di sini adalah bagaimana papan input case uji terlihat seperti: i.stack.imgur.com/qlbH4.png
golffzz


1
@ugoren, saya mempertimbangkan hal itu, tetapi menghitung semua kemungkinan pergerakan berpotensi menghilangkan beberapa pintasan. Saya tidak yakin, mengapa en passant tidak diperlukan: bagian dari poin FEN adalah bahwa ia menyertakan informasi yang cukup untuk membuat en passant mungkin.
Peter Taylor

@ PeterTaylor saya menulis itu hanya untuk mengurangi pekerjaan. Saya heran mengapa tidak ada yang tertarik dengan golf ini :)
golffzz

1
@golffzz tidak boleh input kasus uji Anda menjadi 7k / 8/8/8/8/8 / PP6 / Q1q4K w - - 0 1 bukannya 6pk / 6pp / 8/8/8 / p7 / PP4pp / Q2p2pK w - - 0 1 (yang antara lain, tampaknya memiliki pion pada peringkat akhir)?
Penguino

Jawaban:


8

C - 391 byte

Mengambil input sebagai argumen baris perintah dan mencetak ke stdout dengan kotak berlabel 0 hingga 63.

OK, saya punya beberapa menit jadi saya mencoba untuk menghapus semua bit yang berkaitan dengan deteksi cek. Saya pikir itu sekarang tidak terlalu efisien ...

O(x,y){return((x&7)-(y&7))/5;}
B[64],*b=B,J=32,M,L,x,*X,N;
main(int c,char**V){
for(;x=*V[1]++;c=J&2**V[2])
x>56?*b++=x:x>47?b+=x-48:0;
for(;b-->B;)
for(M=-1,N=*b%J^16,*V=strchr("bGInFJOQrAHkAGHIqAGHIpGHIx",*b|J);*b&&*b&J^c&&(M=M<0?*++*V%J:-M,**V<96);)
for(x=b-B,L=N?9^*b&8:1+(x/8==1+c/6);L--*!(O(x,x+M)|O(x>>3,x+M>>3));L=!*X|~*X&J^c&&N|(!*X^M&1&&M<0^!c)?printf("%d-%d ",b-B,x),L*!*X:0)
X=B+(x+=M);}

478 byte versi pendeteksian

O(x,y){return((x&7)-(y&7))/5;}
B[64],*b=B,c,I,J=32;
main(int C,char**V){
int*D,M,L,t,x,*X,N;
for(;b-B<64;C=c=J&2**V[2])
(t=*V[1]++)>56?*b++=t:t>47?b+=t-48:0;
for(D=b;D-->B;)
for(M=-1,N=*D%J^16,*V=strchr("bGInFJOQrAHkAGHIqAGHIpGHIx",*D|J);*D&&*D&J^C&&(M=M<0?*++*V%J:-M,**V<96);)
for(x=D-B,L=N?9^*D&8:1+(x/8==1+C/6);L--*!(O(x,x+M)|O(x>>3,x+M>>3));L=!*X|~*X&J^C&&N|(!*X^M&1&&M<0^!C)?c^C?I|=*X%J==11:(*X=*D,*D=I=0,main(C^J,V+1),*D=*X,I||printf("%d-%d ",D-B,x)),L*!(*X=t):0)
X=B+(x+=M),t=*X;}

Ini luar biasa, tetapi tidak sesuai dengan pertanyaan: "Abaikan memeriksa, skakmat, dan jalan buntu, raja tidak bisa ditangkap situasi juga."
edc65

@ edc65 Saya tidak setuju dengan interpretasi Anda tentang instruksi yang sayangnya tidak jelas ini. Dikatakan "Keluaran bergerak harus mematuhi aturan pergerakan catur kecuali promosi yang berlalu lalang, kasting, dan gadai." Pindah ke cek tidak muncul dalam daftar tebal aturan untuk diabaikan. Saya menganggap poin-poin ketiga sebagai catatan bahwa Anda tidak perlu mendeteksi situasi-situasi ini dan mencerminkannya dengan cara apa pun dalam output (mis., Konvensi penulisan '+' setelah cek). Juga, test case 2 kedua adalah benar seperti aslinya ditulis, diberikan satu mengabaikan promosi gadai seperti yang ditentukan (tetapi tidak memberikan wawasan tentang pemeriksaan).
feersum

Kasus uji kedua (input) tidak valid memiliki pion hitam di peringkat 8, dan berbeda dari gambar yang diposting sebagai komentar oleh OP ( also here is how test case input boards look like). Mengingat posisi dalam gambar, output kasus uji asli sudah benar sesuai dengan aturan.
edc65

2
Oh, bagus, spesifikasi sebenarnya ada di komentar? Ugh, pertanyaan yang sangat buruk.
feersum

1
Pemenang yang jelas
edc65

2

Java 1455

String q(String f){int[][]b=new int[8][8];int i=0,j=0,k,l,m,n,c;HashSet<String>h=new HashSet<String>();while((c=f.charAt(i))>32){if(c>48&c<57)j+=c-49;if(c==47)j--;if(c>56)b[j%8][j/8]=c;i++;j++;}boolean w=f.charAt(++i)>99;for(i=0;i<8;i++)for(j=0;j<8;j++)if((c=b[i][j])<91?w&c>0:!w){switch(c%32){case 14:for(k=0;k<8;k++){l=(k/4+1)*(k%2*2-1)+i;m=(2-k/4)*(k%4/2*2-1)+j;if(b(l,m)&&(w&b[l][m]%91<40|!w&b[l][m]<91))h.add(h(i,j,l,m));}break;case 11:for(k=0;k<8;k++){l=i+(k==4?1:k/3-1);m=j+(k==4?1:k%3-1);if(b(l,m)&&(w&b[l][m]%91<40|!w&b[l][m]<91))h.add(h(i,j,l,m));}break;case 17:for(k=0;k<8;k++){for(n=1;n<9;n++){l=i+n*(k==4?1:k/3-1);m=j+n*(k==4?1:k%3-1);if(b(l,m)){c=b[l][m];if(w&c%91<40|!w&c<91)h.add(h(i,j,l,m));if(c>0)break;}else break;}}break;case 2:for(k=0;k<4;k++){for(n=1;n<9;n++){l=i+n*(k/2*2-1);m=j+n*(k%2*2-1);if(b(l,m)){c=b[l][m];if(w&c%91<40|!w&c<91)h.add(h(i,j,l,m));if(c>0)break;}else break;}}break;case 18:for(k=0;k<4;k++){for(n=1;n<9;n++){l=i+n*(k/2*(k%2*2-1));m=j+n*((1-k/2)*(k%2*-2+1));if(b(l,m)){c=b[l][m];if(w&c%91<40|!w&c<91)h.add(h(i,j,l,m));if(c>0)break;}else break;}}break;default:m=w?-1:1;if(b[i][j+m]<1){h.add(h(i,j,i,j+m));if(b[i][j+2*m]<1&j==(w?6:1))h.add(h(i,j,i,j+2*m));}for(l=-1;i+l<8&i+l>=0&l<2;l+=2){c=b[i+l][j+m];if(c>0&(c<91?!w:w))h.add(h(i,j,i+l,j+m));}}}return h.toString();}boolean b(int l,int m){return m>=0&m<8&l>=0&l<8;}String h(int i,int j,int l,int m){return""+g(i)+(8-j)+g(l)+(8-m);}char g(int i){return(char)(i+65);}

2

Python 553 649 678

b,Q=raw_input(),range;R=Q(8);D="w"in b
for i in Q(9):b=b.replace(`i`,"_"*i)
if D:b=b.swapcase()
def X(h,v,c):
 h+=x;v+=y
 if c and h|v in R and"a">b[v*9+h]:print chr(65+x)+`8-y`+chr(65+h)+`8-v`;return"_"==b[v*9+h]
for y in R:
 for x in R:
  z=y*9+x;p=b[z];N=p=="n";j=[p in"qrk"]*4+[p in"qbk"]*4
  if"p"==p:j[D]=k=(1,-1)[D];X(1,k,b[z+10*k]<"_");X(-1,k,b[z+8*k]<"_")
  for i in Q(1,(2,(y==(1,6)[D])+2,8)["kp".find(p)]):
   for k in R:j[k]=X((0,0,-i,i,-i,i,-i,i)[k],(i,-i,0,0,-i,-i,i,i)[k],j[k])
  for v,h in((2,1),(1,2)):X(v,h,N);X(-v,-h,N);X(-v,h,N);X(v,-h,N)

Indentasi dua ruang adalah tab char, yang menyimpan 5 byte.

Itu terjadi kepada saya bahwa Anda mungkin dapat membuatnya mengevaluasi langkah-langkah yang masuk akal untuk lapisan yang layak dan menyimpannya di bawah 1024 byte :) Saya mulai mencari melalui pertanyaan lainnya , tetapi sepertinya tidak ada pertanyaan tentang mesin catur codegolf ...


2
Bagus sekali untuk membangkitkan kembali pertanyaan ini! (Saya pikir saya mungkin memberikan hadiah hanya untuk melihat apa yang terjadi.) Jawaban Anda terlihat bagus, tetapi gagal B1C3dan H2H3pada contoh pertama ditunjukkan dalam pertanyaan.
ossifrage pelit

Maaf, bukan H2H3, maksudku G1H3- dengan kata lain, ksatria putihmu hanya belok kiri.
ossifrage pelit

Itu harus menunggu sampai saya menggunakan komputer, tetapi ini adalah bug yang mudah dan saya dapat melihat banyak cara untuk mempersingkatnya juga. Semoga ada entri lain.
Will

@squeamishossifrage diperbaiki :)
Will

1

Python 638 637 (482?) Byte

exec"""p=raw_input()
for x in"12345678":p=p.replace(x,"~"*int(x))
b=map(ord,"#"*21+p[:71].replace("/","##")+"#"*21)
d,e=-10,126
if not"w"in p:b,d=[x^32*(64<x<e)for x in b],10
B=[-9,9,-11,11]
R=[-1,1,-d,d]
Q=B+R
c=Zx:chr(96+x%10)+chr(58-x/10)
for x,p in enumerate(b):
 def O(y):
    if 111<b[y]:print c(x)+c(y)
 s=ZL:[O(x+X)for X in L];m=ZL,z=x:L&(O(z+L[0]),m(L,z+L[0])if e==b[z+L[0]]else m(L[1:]))
 if p==80:e==b[x+d]&(O(x+d)or e==b[x+d*2]&35==b[x-d*2]&O(x+d*2)),111<b[x+d-1]<e&O(x+d-1),111<b[x+d+1]<e&O(x+d+1)
 p==75&s(Q),p==78&s([-12,12,-8,8,-21,21,-19,19]),p==82&m(R),p==66&m(B),p==81&m(Q)""".replace("Z","lambda ").replace("&"," and ")

Catatan: setelah def O(y): ada baris baru dan karakter tab sebelumnyaif

Catatan: dengan menggunakan modul zlib, dimungkinkan untuk mendapatkan kode sumber Python yang valid dari 482 byte hanya dengan mengompresi sumber asli:

#encoding=koi8-r
import zlib
exec zlib.decompress("x°MRKkЦ0>╞~┘Pы Eё╜Е4▌Ц█.9Br*1зБ┤B╠#°■╙=Лoъ╠M⌡│╬г0█\\pcл⌡╝x9╣ЧМф9^:Х╘e:·=м⌠Eй2oЭ╞нЫsQ9─ЩeсS{ЦAR ╕ПЭруюь4жрГыBшОhЖхпy`B▌╬ 58ёt:NхИHшк█╫ЁSK}VBmРПgOyР╢\n+'╬Z║╔▒╣иу√═╢╜-ы#G╙├з▓²Yк=╘л!dуkг≈┴?u$dOФ╘\n▐HфАюВ9]Шж╦╝╦9^┼▄пзИ√ Э│mi╜WeЧa3ъА╗╢бae┘.║WsьdЫ√Ы<ТВэГзьъ
ЙПiB╤≥П-Ъ■⌡<╡▌Б┬1╚3╕лGjщЫЙ(з╧н,>$Eш⌠FыdmШ<x,Р╔Mc;≥м╒2DLc!`Л≥рvЕFCИЪtyв%Н║╞╤≤O╝|'═┤)B|н*┘T╛▐рKпK;╔Я╓АШ&  бУ╗j└;│И╬Ж╝Щ\\4e]P&НРeZ╢5┼ДГt╚У")

1

JavaScript (E6) 481 492 550

Edit Memperbaiki bug jahat di knight moving. Banyak pekerjaan untuk menjaga byte tetap sama.

(Tidak termasuk spasi dan baris baru disimpan agar mudah dibaca)

B=f=>
  [for(c of(o=b='',f=f.split(w=' '))[0])b+=-c?w.repeat(c):c<'0'?z=10:c]+
  [...b].map((c,p)=>
    (K=d=>(d=b[p+d])==w?0:d>'a'?m:d>'A'?-m:9)(0)-1||(
      t='pPkKnNrRQqBb'.search(c),
      O=d=>K(d)<1?o+=w+P(p)+P(p+d):0,
      S=(d,s=0)=>O(d*++s)&&!K(d*s)&&S(d,s),
      //o+='\n'+P(p)+' '+c, // Uncomment to display pieces list
      t<2?[for(i of[~(s=t<1?z:-z),1-s])~K(i)||O(i)]+!K(s)&&O(s)&&(p/z|0)==t*5+1&!K(s+=s)&&O(s)
      :t<6?[for(i of t<4?[1,9,z,11]:[12,8,21,19])O(i)+O(-i)]
      :[for(i of[t>7&&9,t>7&&11,t<z,t<z&&z])S(i)+S(-i)]
    )
  ,m=f[1]<'w'?1:-1,
  //console.log(','+([...b]+',').replace(/1,0/g,'\n')), // Uncomment to display chessboard
  P=p=>'ABCDEFGH'[p%z]+(9+~p/z|0))&&o

Kurang Golf

B=f=>(
  o=b='',[for(c of f)b+=-c?'.'.repeat(c):c],
  m=(w=b[72]=='w')?1:2,n=3-m,
  t=0,
  P=p=>'ABCDEFGH'[p%9]+(9+~p/9|0),
  b=b.slice(0,71),
  // console.log(b.replace(/\//g,'\n')), // Uncomment to display chessboard

  [...b].map((c,p)=>{
    r=p/9|0
    K=k=>(k=b[k])=='.'?0:k>'a'?m:k>'A'?n:9
    J=d=>K(p+d)<2,
    O=d=>J(d)?o+=' '+P(p)+P(p+d):0,
    S=(s,d)=>O(d*++s)&&!K(p+d*s)?S(s,d):0;

    if(K(p)==2){
      // o+='\n'+P(p)+ ' '+c; // Uncomment to display pieces list
      if (c=='P')
      {
        (f=!K(p-9))&&O(-9),
        f&r==6&&!K(p-18)&&O(-18),
        [for(i of[10,8])K(p-i)==1&&O(-i)]
      }
      else if (c=='p')
      {
        (f=!K(p+9))&&O(+9),
        f&r==1&&!K(p+18)&&O(+18),
        [for(i of[10,8])K(p+i)==1&&O(+i)]
      }
      else if (c=='K' |c=='k')
        [for(i of[1,8,9,10])O(i)+O(-i)]
      else if (c=='N' | c=='n')
        [for(i of[11,7,19,17])O(i)+O(-i)]
      else 
      {
        if (c!='r' & c!='R')
          [for(i of[10,8])S(0,i)+S(0,-i)]
        if (c!='b' & c!='B')
          [for(i of[9,1])S(0,i)+S(0,-i)]
      }
    }     
  }),
  o
)

Tes di konsol FireFox / FireBug

B("7k/8/8/8/8/1P6/P7/Q1q4K w - - 0 1")

Keluaran

B3B4 A2A3 A2A4 A1B2 A1C3 A1D4 A1E5 A1F6 A1G7 A1H8 A1B1 A1C1 H1G1 H1H2 H1G2

1

JAVA 631 599 594

Memperbaiki bug dalam versi 599-byte (terima kasih Jack telah menunjukkan ini!) Dan mempersingkat kode menjadi 594-byte.

class F{
public static void main(String[]A){int p=0,i=8,u,v,d,z[]={0,1,-1,2,1,0,1};String f=A[0],S[]="p,n,rqk,bqk,aA,PNBRQK,aAPNBRQK".split(",");
for(;i>0;)f=f.replace(""+i,"a"+(--i==0?"":i));
for(;p<448;p++)
for(int k=p%7,w=A[1].equals("w")?32:0,c=f.charAt(p/7%8+p/56*9),a=z[k],b=k==4?2:1,q=0;S[(k/2-1|k-1)/2].indexOf(c+w)>=0&&q++<(k<3?1:4);i=a,a=b,b=-i){
for(i=1,d=97;d==97&&((u=p/7%8+i*a)|(v=p/56+i*b*(1-w/16)))>=0&&(u|v)<8&&i<(k>4?(c=='K'||c=='k')?2:9:(k<1&&p/56==w/6+1?3:2))&&S[61/(61^(k*k-2))+5].indexOf((d=f.charAt(u+9*v))-w)>=0;i++)System.out.printf("%c%d%c%d ",65+p/7%8,8-p/56,65+u,8-v);}}}

Kompilasi: javac F.java
Jalankan: java F 6pk/6pp/8/8/8/p7/PP4pp/Q2p2pK w - - 0 1
Output:B2B3 B2B4 B2A3 A1B1 A1C1 A1D1 H1H2 H1G1 H1G2


Karena 3Q4/p4r1k/P4pp1/4P3/5n2/3P4/4BbbP/RN3KN1 w - - 0 0aku tidak melihat gerakan F1F2atau F1G2apakah raja mampu menangkap?
Jack

Terima kasih telah menunjukkan ini. Kamu benar. Versi yang diterbitkan ini sudah dioptimalkan :(
Bob Genom

Jadi saya harus kembali ke solusi panjang 631 byte saya.
Bob Genom

Dengan menggunakan situs kami, Anda mengakui telah membaca dan memahami Kebijakan Cookie dan Kebijakan Privasi kami.
Licensed under cc by-sa 3.0 with attribution required.