forked from tomhanika/conexp-clj
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathimplication-closure.clj
95 lines (82 loc) · 3.32 KB
/
implication-closure.clj
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
;; Daniel Borchmann, 2010
;; This file is in the public domain.
;; A simple macro that computes a special context from a given set of
;; implications (it's the standard context of the closure system of
;; the given implications). Use it with
;;
;; (closure-context #{1 2 3 4}
;; 1 2 -> 3 --
;; 2 -> 4 --
;; 1 3 4 -> 2)
;;
;; It will give you something like
;;
;; |1 2 3 4
;; ---------+--------
;; #{1 3} |x . x .
;; #{1 4} |x . . x
;; #{2 4} |. x . x
;; #{3 4} |. . x x
;; #{2 3 4} |. x x x
;;
;; Note that -- is used to separate the implication descriptions and
;; that #{..} is Clojure's notation for sets. This file also contains
;; implementations for {intersection,union}-of-closure-system.
(require 'conexp.main)
(in-ns 'conexp.main)
;;
(defn- implications-from-macro
"Implements the syntax used by closure-context."
[implication-specification]
(let [impl-blocks (remove #(= '-- (first %))
(partition-by #(= '-- %)
implication-specification))]
(vec (for [impl-spec impl-blocks]
(let [parts (partition-by #(= '-> %) impl-spec)]
(when-not (= 3 (count parts))
(illegal-argument (str "Malformed implication specification: "
impl-spec ".")))
`(make-implication ~(set (nth parts 0))
~(set (nth parts 2))))))))
(defmacro closure-context
"For a given collection of implications and a base set returns the
full ordinal scala of the lattice of all closed subsets of the base
set with respect to the given implications. Implications are written
as shown:
(closure-context #{1 2 3 4}
1 2 3 -> 4 --
3 4 -> 1 --
1 4 -> 2)
Note that the first argument is the base set. After that the
implications are written as <elements> -> <elements>, separated by a
--. Note that the elements can be everything except the symbols ->
and -- (of course)."
[base-set & implications]
(let [implications (implications-from-macro implications)]
`(do
(let [impls# ~implications]
(when-not (forall [impl# impls#]
(and (subset? (premise impl#) ~base-set)
(subset? (conclusion impl#) ~base-set)))
(illegal-argument "Given implications are not compatible with "
"given base-set."))
(context-from-clop ~base-set (clop-by-implications impls#))))))
;;;
(defn intersection-of-closure-systems
"Given two contexts ctx-1 and ctx-2 with the same object set
computes a context for the intersection of the concept lattices of
both."
[ctx-1 ctx-2]
(assert (= (objects ctx-1) (objects ctx-2)))
(context-from-clop (objects ctx-1)
(clop-by-implications (union (stem-base (dual-context ctx-1))
(stem-base (dual-context ctx-2))))))
(defn union-of-closure-systems
"Given two contexts ctx-1 and ctx-2 with the same object set
computes a context for the smalles lattice containing the union of
both concept lattices."
[ctx-1 ctx-2]
(assert (= (objects ctx-1) (objects ctx-2)))
(context-apposition ctx-1 ctx-2))
;;;
nil