Sto modellando un albero genealogico con core.logic. Vorrei rispondere alle domande con run*
e restituirle tutti i risultati senza duplicare. Sostituire tutto defn
con def tabled
mi dà i risultati che mi aspetto (almeno per ora) e so che condu
e onceo
possono ridurre il numero di risultati, ma non sono sicuro se uno di questi sia il modo migliore per eliminare i duplicati.Eliminazione dei risultati duplicati durante l'interrogazione di un albero genealogico con core.logic
Sono particolarmente preoccupato per il mio approccio attuale perché sembra un lavoro duplicato per dichiarare sia le relazioni che le funzioni. So che alcune delle mie relazioni sono "reciprocamente ricorsive" (mothero
e womano
di riferimento l'un l'altro), ma l'ho fatto perché in futuro potrei aggiungere un nuovo (defrel mother*)
, che dovrebbe consentire di dedurre che una madre è sia un genitore che una donna
(defrel man* person)
(defrel woman* person)
(defrel parent* child father)
(fact man* :Father)
(fact woman* :Mother)
(fact man* :Son)
(fact woman* :Daughter)
(fact parent* :Son :Father)
(fact parent* :Son :Mother)
(fact parent* :Daughter :Father)
(fact parent* :Daughter :Mother)
(defn mano [person]
(conde
[(man* person)]
[(fresh [c]
(fathero c person))]))
(defn womano [person]
(conde
[(woman* person)]
[(fresh [c]
(mothero c person))]))
(defn parento [child person]
(conde
[(parent* child person)]
[(mothero child person)]
[(fathero child person)]))
(defn fathero [child father]
(all
(mano father)
(parento child father)))
(defn mothero [child mother]
(all
(womano mother)
(parento child mother)))
(defn siblingso [c1 c2 mother father]
(all
(mothero c1 mother)
(mothero c2 mother)
(fathero c1 father)
(fathero c2 father)
(!= c1 c2)))
(run 10 [q]
(fresh [child parent]
(parento child parent)
(== q [child parent])))
(run 10 [q]
(fresh [c1 c2 p1 p2]
(siblingso c1 c2 p1 p2)
(== q [c1 c2 p1 p2])))
Il motivo per cui ho definito le funzioni parento, fathero, siblingso, ecc. è che poi voglio usarli come elementi costitutivi per definire relazioni più grandi. Come cousinso sarebbe più facile definire usando siblingso e parento. invece di limitarlo a solo genitore *. Mi piacerebbe finire per costruire funzioni per qualsiasi tipo di relazione, fratelli, zii, 2 ° cugino, antenati, discendenti, imparentati, ecc. Io non so come definire questi in modo che accetterebbero nuove relazioni come sister * mentre termina anche quando viene eseguito con run *. – WuHoUnited
Ho aggiornato la mia risposta per includere un esempio di (versione semplificata) di 'brothero', spero che questo ti aiuti a raggiungere i tuoi obiettivi! ;-) –
Ho fatto un'altra piccola modifica nell'ultimo esempio, in quanto non è stato necessario utilizzare 'defne' in questo caso. –