![]() |
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
![]() ![]() ![]() ![]() ![]() ![]() ![]() ![]() |
|
XOR | ||
x | 0 | 1 |
0 | 0 | 1 |
1 | 1 | 0 |
Pokud prázdné pole a černé i bílé kameny reprezentujeme hodnotami 0 až 12, zkažené rošády jedním číslem s nastavenými spodními 4 bity (malá bílá, velká bílá, malá černá a velká černá rošáda, proto 4) a právo braní mimochodem číslem od 0 do 8 podle sloupce, kde lze brát, a 0 znamená, že brát nelze, může být výpočet takto jednoduchý:
u32 nahodaSachovnice[64][13];/* náhodná čísla pro kameny na šachovnici */ u32 nahodaZkazeneRosady[16]; /* celkem 2^(počet rošád) = 2^4 = 16 možností */ u32 nahodaMimochodem[9]; /* 8 sloupců + 1 (nelze brát) = 9 možností braní mimochodem */ u32 nahodaBily; /* kdo je na tahu */ u32 F(const u8 *sachovnice, int rosady, int mimochodem, int bily) { u32 vysledek = 0; int i; for (i = 0; i < 64; i++) { vysledek ^= nahodaSachovnice[i][sachovnice[i]]; } vysledek ^= nahodaZkazeneRosady[rosady]; vysledek ^= nahodaMimochodem[mimochodem]; if (bily) vysledek ^= nahodaBily; return vysledek; }
Díky funkci XOR výsledek závisí na všech polích šachovnice, drobná změna na jediném poli nebo v jiné vlastnosti pozice (braní mimochodem a podobně) způsobí, že výsledek bude vyxorován s jedním úplně jiným číslem a tedy úplně jiný. Díky tomu bude funkce F rozhazovat hodnoty pozic rovnoměrně do celé haš tabulky a kontrolní funkce G používající jinou sadu náhodných čísel bude mít úplně jiné kolize. Myslím, že idea hašování objektu pomocí náhodných čísel a funkce XOR je zajímavá a mohla by se v nějaké zcela odlišné úloze hodit i programátorům, kteří se nikdy šachům věnovat nebudou.
Funkce F a G už jsou definovány správně, ale máme ještě jeden problém a tím je efektivita výpočtu. Funkci budeme volat v každém navštíveném uzlu stromu propočtu včetně listů. Musíme ušetřit každou instrukci. Bohužel ve výpočtu je cyklus přes celou šachovnici a to si nemůžeme dovolit. Naštěstí máme funkce definované pomocí XOR a to je asociativní tj. (A XOR B) XOR C = A XOR (B XOR C), komutativní tj. A XOR B = B XOR A a navíc platí A XOR B XOR B = A. Díky tomu může funkci počítat i inkrementálně. To v podstatě znamená, že podle výše uvedeného kusu kódu spočítáme F a G jen jednou a to na začátku propočtu. Dejme tomu, že program má vymyslet první tah bílého ze základního postavení. Spočítá si tedy F a G pro základní postavení a dejme tomu, že v propočtu zkoumá variantu 1. Jf3 tedy tah jezdcem z g1 na f3. Udělá to následovně:
vysledek = hodnota_F_pro_zakladni_postaveni; vysledek ^= nahodaSachovnice[G1][PRAZDNE_POLE]; vysledek ^= nahodaSachovnice[G1][BILY_JEZDEC]; vysledek ^= nahodaSachovnice[F3][PRAZDNE_POLE]; vysledek ^= nahodaSachovnice[F3][BILY_JEZDEC]; vysledek ^= nahodaBily;
Tedy zaxorují se jenom odlišnosti. Díky tomu, že XOR je samo vůči sobě inverzní operací, nemusíme ani myslet na to, která vlastnost tahem Jf3 nastává a která se ruší. Kdybychom funkce F a G místo na XOR vystavěli například na + (+ tak jak se běžně počítá na procesoru tj. modulová aritmetika se zahazováním bitů které se nevejdou do typu výsledku), museli bychom rušené jevy (zde Jg1 a nic na f3) místo sčítání odečítat a nastávající (zde nic na g1 a Jf3) přičítat. Zároveň je vidět, že bychom mohli náš výpočet s XORem trochu urychlit, pokud bychom pro prázdná políčka místo náhodných čísel použili natvrdo nuly. V příkladu s tahem Jf3 bychom místo s 5 XORy vystačili se třemi.
Příští díl bude o něco techničtější. Řekneme si něco o tom, jak reprezentovat pozici, jak generovat a ukládat tahy a podobně. To všechno jsou věci, které by každý zkušený programátor jistě nějak zvládl, ale existuje celá řada technik a drobných fíglů, které nejsou zcela bezprostřední a mohou ušetřit hodně času jak programátorovi, tak i jeho šachovému programu.
|
||
KOMENTARZE
Nie ma komentarzy dla tej pozycji. |
||
Tylko zarejestrowani użytkownicy mogą dopisywać komentarze.
|
1. |
Pacman linux Download: 5004x |
2. |
FreeBSD Download: 9214x |
3. |
PCLinuxOS-2010 Download: 8700x |
4. |
alcolix Download: 11096x |
5. |
Onebase Linux Download: 9809x |
6. |
Novell Linux Desktop Download: 0x |
7. |
KateOS Download: 6372x |
1. |
xinetd Download: 2535x |
2. |
RDGS Download: 937x |
3. |
spkg Download: 5040x |
4. |
LinPacker Download: 10209x |
5. |
VFU File Manager Download: 3311x |
6. |
LeftHand Mała Księgowość Download: 7340x |
7. |
MISU pyFotoResize Download: 2973x |
8. |
Lefthand CRM Download: 3673x |
9. |
MetadataExtractor Download: 0x |
10. |
RCP100 Download: 3270x |
11. |
Predaj softveru Download: 0x |
12. |
MSH Free Autoresponder Download: 0x |