Add some algo

This commit is contained in:
Loic GUEGAN 2019-01-27 19:34:56 +01:00
parent 35bc8cbb15
commit ee5bce70f5
3 changed files with 110 additions and 0 deletions

30
src/quick-find.lisp Normal file
View file

@ -0,0 +1,30 @@
;; Create network nodes
(defun create-network (n)
"Build a quick-find network using a dynamic vector"
(let ((nodes (make-array n :fill-pointer 0)))
(dotimes (id n)
(vector-push id nodes))
nodes))
;; Check if two nodes are connected
(defmacro connected (network n1 n2)
" Return t if there is a path between n1 and n2, nil otherwise. connected represent the find operation of the Quick Find Algorithm"
`(= (elt ,network ,n1) (elt ,network ,n2)))
;; Link two nodes in the network
(defun union_ (network n1 n2)
"Link two nodes in the quick-find network. union_ represent the union operation of the Quick Find Algorithm"
(let ((v-n1 (elt network n1))
(v-n2 (elt network n2))
(new-network (copy-seq network)))
(dotimes (n (length new-network))
(if (= (elt new-network n) v-n2) (setf (elt new-network n) v-n1)))
new-network))
;; Union consing version
(defmacro nunion_ (network n1 n2)
"A cosing version of union_"
`(setq ,network (union_ ,network ,n1 ,n2)))

33
src/quick-union.lisp Normal file
View file

@ -0,0 +1,33 @@
;; Create network nodes
(defun create-network (n)
"Build a quick-find network using a dynamic vector"
(let ((nodes (make-array n :fill-pointer 0)))
(dotimes (id n)
(vector-push id nodes))
nodes))
;; Find the root of a node
(defun find-root (network node)
"Find the root of a sub-tree in the network."
(do ((id node value)
(value (elt network node) (elt network value)))
((= id value) id)))
;; Check if two nodes are connected
(defmacro connected (network n1 n2)
"Return true if n1 and n2 are connected and nil otherwise. connection represent
the find operation on the Quick Union algorithm"
`(= (find-root ,network ,n1) (find-root ,network ,n2)))
;; Link two nodes together
(defun union_ (network n1 n2)
"Connect to sub-tree together. union represent the union operation on the Quick Union algorithm"
(let ((new-network (copy-seq network)))
(setf (elt new-network (find-root new-network n1))
(find-root new-network n2))
new-network))
;; A consed version of union_
(defmacro nunion_ (network n1 n2)
`(setf ,network (union_ ,network ,n1 ,n2)))

View file

@ -0,0 +1,47 @@
;; Create network nodes: A two dimensionnal array.
;; 1st dimension = the network
;; 2nd dimension = each subtree node quantities
(defun create-network (n)
"Build a quick-find network using a multi-dimensional dynamic vector"
(let ((network (make-array `(2 ,n) :initial-element 1)))
(dotimes (id n)
(setf (aref network 0 id) id))
network))
;; Find the root of a node
(defun find-root (network node)
"Find the root of a sub-tree in the network."
(do ((id node value)
(value (aref network 0 node) (aref network 0 value)))
((= id value) id)))
;; Check if two nodes are connected
(defmacro connected (network n1 n2)
"Return true if n1 and n2 are connected and nil otherwise. connection represent
the find operation on the Quick Union algorithm"
`(= (find-root ,network ,n1) (find-root ,network ,n2)))
;; Link two nodes together
(defun union_ (network n1 n2)
"Connect to sub-tree together. union represent the union operation on the Quick Union algorithm"
(let ((new-network (copy-tree network))) ; Duplicate network
(let* ((n1-root (find-root new-network n1))
(n2-root (find-root new-network n2))
(n1-size (aref new-network 1 n1-root))
(n2-size (aref new-network 1 n2-root)))
(if (>= n1-size n2-size) ; Check which subtree is LARGER (not deeper)
(progn (setf (aref new-network 0 n2-root) (aref new-network 0 n1-root)) ; Modify the second node
(setf (aref new-network 1 n1-root) ; Update tree larger
(+ (aref new-network 1 n1-root) (aref new-network 1 n2-root))))
(progn (setf (aref new-network 0 n1-root) (aref new-network 0 n2-root)) ; Otherwise modify the first node
(setf (aref new-network 1 n2-root) ; Update tree larger
(+ (aref new-network 1 n2-root) (aref new-network 1 n1-root)))))
new-network)))
;; A consed version of union_
(defmacro nunion_ (network n1 n2)
`(setf ,network (union_ ,network ,n1 ,n2)))