Skip to content

Latest commit

 

History

History
 
 

cv04

Folders and files

NameName
Last commit message
Last commit date

parent directory

..
 
 

Cvičenie 4

Riešenie odovzdávajte podľa pokynov na konci tohoto zadania do utorka 22.3. 23:59:59.

Súbory potrebné pre toto cvičenie si môžete stiahnuť ako jeden zip cv04.zip.

Logická hádanka (3b)

Nasledujúcu logickú hádanku zakódujte do výrokovej logiky a vyriešte pomocou SAT solvera. Odovzdávajte program, ktorý vygeneruje vstup pre SAT solver, pustí SAT solver, načíta a dekóduje výstup a vypíše riešenie. V programe (v komentároch) uveďte formuly, ktorými ste problém popísali.

Pán a pani Smithovci a ich dve deti sú typická americká rodina. Vieme, že práve dve tvrdenia z nasledovných sú pravdivé:

  • George a Dorothy sú pokrvní príbuzní
  • Howard je starší než George
  • Virginia je mladšia než Howard
  • Virginia je staršia než Dorothy

Aké je krstné meno otca, mamy, syna, dcéry?

Ako prvú vec v tejto úlohe samozrejme musíme vymyslieť, aké výroky budeme používať. Sú ale dve základné možnosti: korektnejšialenivejšia.

V korektnejšej použijeme okrem tvrdení typu „X je otec“, „Y je dcéra“ atď. aj tvrdenia „X a Y sú pokrvní príbuzní“, „X je starší(-ia) ako Y“ a korektne vyjadríme vzťahy medzi nimi (kto s kým je / môže byť / nesmie byť v „typickej americkej rodine“ pokrvný príbuzný (starší, mladší...).

lenivejšom riešení použijeme iba prvý druh tvrdení a vyjadríme naše tvrdenia už priamo pomocou nich. Tým sme ale kus práce spravili my (a nenechali ho na SAT solver) a nezapísali sme presne tvrdenia, ktoré sme mali (a mohli sme sa ľahko pomýliť pri ich prepise). Takéto riešenia tejto úlohy budeme síce akceptovať, ale dostanú maximálne 2 body.

Pomôcka 1: Hádanka má práve jedno riešenie. SAT solver môže nájsť viacero modelov vašej teórie, všetky by však mali rovnako ohodnocovať premenné, ktoré kódujú riešenie a mali by sa líšiť len ohodnotením pomocných premenných.

Pomôcka 2: Okrem explicitne vyslovených podmienok máme aj nejaké nevyslovené, ktoré sa skrývajú pod pojmom „typická americká rodina“. Čo všetko pod tým autor hádanky myslel?

Pomôcka 3: Ďalšie nevyslovené podmienky, ktoré pri formalizácii musíme uvažovať, sú typu: Tá istá osoba nemôže byť zároveň otcom aj synom.

Pomôcka 4: Zadefinujte obmedzenia na členov rodiny pre pojmy: pokrvní príbuzní, starší/mladší. Toto úzko súvisí s ujasnením si skutočností v pomôcke 2. Ktorí členovia rodiny môžu byž pokrvní príbuzní, ktorý člen môže byť starší od ktorého, …?

Pomôcka 5: Keby sme iba jednoducho zakódovali všetky tvrdenia a dali na vstup SAT solveru, tak by našiel možnosti, kedy sú všetky naraz pravdivé. Úloha ale hovorí, že práve dve tvrdenia majú byť pravdivé (a dve teda nepravdivé). Toto zabezpečíme tak, že použijeme 4 pomocné premenné (jednu pre každé tvrdenie), ktoré budú pravdivé práve vtedy, keď dotyčné tvrdenie je pravdivé. Čiže na vstup SAT solveru chceme dať formuly tvaru p1 ↔︎ TVRDENIE1, p2 ↔︎ TVRDENIE2, … (toto budú samozrejme zakaždým 2 implikácie, ktoré sa môžu skonvertovať na viacero CNF kláuz) a potom formuly, ktoré zabezpečia, že práve 2 tvrdenia sú pravdivé a 2 nepravdivé.

Pomôcka 6: Zabezpečiť, že práve 2 tvrdenia sú pravdivé, sa dá napríklad vymenovaním všetkých možností, čo ale bude DNF, z ktorého roznásobením vznikne veľa kláuz. Lepšia možnosť je povedať, že aspoň 2 musia byť pravdivé a nesmie byť viac ako 2 pravdivých (t.j. nesmú byť 3 pravdivé). Jednotlivé možnosti v druhej podmienke sú negácie konjunkcií, čo sú priamo klauzy. Prvá podmienka bude ešte stále DNF, ale iba s konjunkciami s 2 premennými. Dá sa to však tiež vylepšiť, ak ju preformulujeme ako „nesmú byť viac ako 2 (aspoň 3) nepravdivé“.

Technické detaily riešenia

Riešenie odovzdajte do vetvy cv04 v adresári cv04.

Odovzdajte súbor family.py/family.cpp/Family.java, ktorý zakóduje problém, pustí SAT solver, načíta a dekóduje výstup a vypíše na štandardný výstup riešenie vo formáte:

otec: Oooo
matka: Mmmm
syn: Ssss
dcera: Dddd

Ak chcete v pythone použiť knižnicu z examples/sat, nemusíte si ju kopírovať do aktuálne adresára, stačí ak na začiatok svojej knižnice pridáte:

import os
import sys
sys.path[0:0] = [os.path.join(sys.path[0], '../examples/sat')]
import sat

Odovzdávanie riešení v iných jazykoch konzultujte s cvičiacimi.