Mercurial > dylan
view sicm/bk/utils.clj @ 9:23db8b1f0ee7
Softened tone in science minus science.
author | Dylan Holmes <ocsenave@gmail.com> |
---|---|
date | Sat, 29 Oct 2011 21:18:54 -0500 |
parents | b4de894a1e2e |
children |
line wrap: on
line source
2 (ns sicm.utils)4 ;***** GENERIC ARITHMETIC5 (ns sicm.utils)6 (in-ns 'sicm.utils)8 (defprotocol Arithmetic9 (zero [this])10 (one [this]))13 (extend-protocol Arithmetic14 java.lang.Number15 (zero [this] 0)16 (one [this] 1))18 (extend-protocol Arithmetic19 clojure.lang.Seqable20 (zero [this] (map zero this))21 (one [this] (map one this)))24 ;***** TUPLES AND MATRICES25 (in-ns 'sicm.utils)27 (defprotocol Spinning28 (up? [this])29 (down? [this]))31 (defn spin32 "Returns the spin of the Spinning s, either :up or :down"33 [#^Spinning s]34 (cond (up? s) :up (down? s) :down))37 (deftype Tuple38 [spin coll]39 clojure.lang.Seqable40 (seq [this] (seq (.coll this)))41 clojure.lang.Counted42 (count [this] (count (.coll this))))44 (extend-type Tuple45 Spinning46 (up? [this] (= ::up (.spin this)))47 (down? [this] (= ::down (.spin this))))49 (defmethod print-method Tuple50 [o w]51 (print-simple (str (if (up? o) 'u 'd) (.coll o)) w))55 (defn up56 "Create a new up-tuple containing the contents of coll."57 [coll]58 (Tuple. ::up coll))60 (defn down61 "Create a new down-tuple containing the contents of coll."62 [coll]63 (Tuple. ::down coll))66 (in-ns 'sicm.utils)68 (defn numbers?69 "Returns true if all arguments are numbers, else false."70 [& xs]71 (every? number? xs))73 (defn contractible?74 "Returns true if the tuples a and b are compatible for contraction,75 else false. Tuples are compatible if they have the same number of76 components, they have opposite spins, and their elements are77 pairwise-compatible."78 [a b]79 (and80 (isa? (type a) Tuple)81 (isa? (type b) Tuple)82 (= (count a) (count b))83 (not= (spin a) (spin b))85 (not-any? false?86 (map #(or87 (numbers? %1 %2)88 (contractible? %1 %2))89 a b))))93 (defn contract94 "Contracts two tuples, returning the sum of the95 products of the corresponding items. Contraction is recursive on96 nested tuples."97 [a b]98 (if (not (contractible? a b))99 (throw100 (Exception. "Not compatible for contraction."))101 (reduce +102 (map103 (fn [x y]104 (if (numbers? x y)105 (* x y)106 (contract x y)))107 a b))))109 ;***** MATRICES110 (in-ns 'sicm.utils)111 (require 'incanter.core) ;; use incanter's fast matrices113 (defprotocol Matrix114 (rows [this])115 (cols [this])116 (diagonal [this])117 (trace [this])118 (determinant [this]))120 (extend-protocol Matrix121 incanter.Matrix122 (rows [this] (map down this)))127 (defn count-rows [matrix]128 ((comp count rows) matrix))130 (defn count-cols [matrix]131 ((comp count cols) matrix))134 (defn matrix-by-rows135 "Define a matrix by giving its rows."136 [& rows]137 (cond138 (not (all-equal? (map count rows)))139 (throw (Exception. "All rows in a matrix must have the same number of elements."))140 :else141 (reify Matrix142 (rows [this] (map down rows))143 (cols [this] (map up (apply map vector rows)))144 (diagonal [this] (map-indexed (fn [i row] (nth row i) rows)))145 (trace [this]146 (if (not= (count-rows this) (count-cols this))147 (throw (Exception.148 "Cannot take the trace of a non-square matrix."))149 (reduce + (diagonal this))))151 (determinant [this]152 (if (not= (count-rows this) (count-cols this))153 (throw (Exception.154 "Cannot take the determinant of a non-square matrix."))155 (reduce * (diagonal this))))156 )))159 (defn matrix-by-cols160 "Define a matrix by giving its columns."161 [& cols]162 (cond163 (not (all-equal? (map count cols)))164 (throw (Exception. "All columns in a matrix must have the same number of elements."))165 :else166 (reify Matrix167 (cols [this] (map up cols))168 (rows [this] (map down (apply map vector cols)))169 (diagonal [this] (map-indexed (fn [i col] (nth col i) cols)))170 (trace [this]171 (if (not= (count-cols this) (count-rows this))172 (throw (Exception.173 "Cannot take the trace of a non-square matrix."))174 (reduce + (diagonal this))))176 (determinant [this]177 (if (not= (count-cols this) (count-rows this))178 (throw (Exception.179 "Cannot take the determinant of a non-square matrix."))180 (reduce * (map-indexed (fn [i col] (nth col i)) cols))))181 )))183 (extend-protocol Matrix Tuple184 (rows [this] (if (down? this)185 (list this)186 (map (comp up vector) this)))188 (cols [this] (if (up? this)189 (list this)190 (map (comp down vector) this))))192 (defn matrix-multiply [A B]193 (apply matrix-by-rows194 (for [a (rows A)]195 (for [b (cols B)]196 (contract a b)))))