001/**
002 *                    BioJava development code
003 *
004 * This code may be freely distributed and modified under the
005 * terms of the GNU Lesser General Public Licence.  This should
006 * be distributed with the code.  If you do not have a copy,
007 * see:
008 *
009 *      http://www.gnu.org/copyleft/lesser.html
010 *
011 * Copyright for this code is held jointly by the individual
012 * authors.  These should be listed in @author doc comments.
013 *
014 * For more information on the BioJava project and its aims,
015 * or to join the biojava-l mailing list, visit the home page
016 * at:
017 *
018 *      http://www.biojava.org/
019 *
020 * Created on 5 Mar 2013
021 * Created by Andreas Prlic
022 *
023 * @since 3.0.6
024 */
025package org.biojava.nbio.structure.math;
026
027import java.io.Serializable;
028
029
030/**
031 *
032 *  A sparse vector, implemented using a symbol table.
033 *
034 *  Derived from http://introcs.cs.princeton.edu/java/44st/SparseVector.java.html
035 *
036 *  For additional documentation, see <a href="http://introcs.cs.princeton.edu/44st">Section 4.4</a> of
037 *  <i>Introduction to Programming in Java: An Interdisciplinary Approach</i> by Robert Sedgewick and Kevin Wayne.
038 */
039
040public class SparseVector implements Serializable{
041        /**
042         *
043         */
044        private static final long serialVersionUID = 1174668523213431927L;
045
046        private final int N;             // length
047
048        private SymbolTable<Integer, Double> symbolTable;  // the vector, represented by index-value pairs
049
050
051        /** Constructor. initialize the all 0s vector of length N
052         *
053         * @param N
054         */
055        public SparseVector(int N) {
056                this.N  = N;
057                this.symbolTable = new SymbolTable<>();
058        }
059
060        /** Setter method (should it be renamed to set?)
061        *
062        * @param i set symbolTable[i]
063        * @param value
064        */
065        public void put(int i, double value) {
066                if (i < 0 || i >= N) throw new IllegalArgumentException("Illegal index " + i + " should be > 0 and < " + N);
067                if (value == 0.0) symbolTable.delete(i);
068                else              symbolTable.put(i, value);
069        }
070
071        /** get a value
072         *
073         * @param i
074         * @return  return symbolTable[i]
075         */
076        public double get(int i) {
077                if (i < 0 || i >= N) throw new IllegalArgumentException("Illegal index " + i + " should be > 0 and < " + N);
078                if (symbolTable.contains(i)) return symbolTable.get(i);
079                else                return 0.0;
080        }
081
082        // return the number of nonzero entries
083        public int nnz() {
084                return symbolTable.size();
085        }
086
087        // return the size of the vector
088        public int size() {
089                return N;
090        }
091
092        /** Calculates the dot product of this vector a with b
093         *
094         * @param b
095         * @return
096         */
097        public double dot(SparseVector b) {
098                SparseVector a = this;
099                if (a.N != b.N) throw new IllegalArgumentException("Vector lengths disagree. " + a.N + " != " + b.N);
100                double sum = 0.0;
101
102                // iterate over the vector with the fewest nonzeros
103                if (a.symbolTable.size() <= b.symbolTable.size()) {
104                        for (int i : a.symbolTable)
105                                if (b.symbolTable.contains(i)) sum += a.get(i) * b.get(i);
106                }
107                else  {
108                        for (int i : b.symbolTable)
109                                if (a.symbolTable.contains(i)) sum += a.get(i) * b.get(i);
110                }
111                return sum;
112        }
113
114        /** Calculates the 2-norm
115         *
116         * @return
117         */
118        public double norm() {
119                SparseVector a = this;
120                return Math.sqrt(a.dot(a));
121        }
122
123        /** Calculates  alpha * a
124         *
125         * @param alpha
126         * @return
127         */
128        public SparseVector scale(double alpha) {
129                SparseVector a = this;
130                SparseVector c = new SparseVector(N);
131                for (int i : a.symbolTable) c.put(i, alpha * a.get(i));
132                return c;
133        }
134
135        /** Calcualtes return a + b
136         *
137         * @param b
138         * @return
139         */
140        public SparseVector plus(SparseVector b) {
141                SparseVector a = this;
142                if (a.N != b.N) throw new IllegalArgumentException("Vector lengths disagree : " + a.N + " != " + b.N);
143                SparseVector c = new SparseVector(N);
144                for (int i : a.symbolTable) c.put(i, a.get(i));                // c = a
145                for (int i : b.symbolTable) c.put(i, b.get(i) + c.get(i));     // c = c + b
146                return c;
147        }
148
149        @Override
150        public String toString() {
151                StringBuilder s = new StringBuilder();
152                for (int i : symbolTable) {
153                        s.append("(");
154                        s.append(i);
155                        s.append(", ");
156                        s.append(symbolTable.get(i));
157                        s.append(") ");
158                }
159                return s.toString();
160        }
161
162
163}
164