Changeset 10108 for branches/HeuristicLab.Analysis.AlgorithmBehavior/HeuristicLab.Analysis.AlgorithmBehavior.Analyzers
- Timestamp:
- 11/05/13 23:04:16 (11 years ago)
- File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
branches/HeuristicLab.Analysis.AlgorithmBehavior/HeuristicLab.Analysis.AlgorithmBehavior.Analyzers/3.3/DistanceMatrixToPoints.cs
r10098 r10108 72 72 } 73 73 74 public static double[][] MetricMDS(double[][] dm) { 74 //based on R's cmdscale 75 public static double[][] MetricMDS(double[][] dm, int k = 2, bool add = false) { 75 76 double[][] points = new double[dm.Length][]; 76 double[,] a = new double[dm.Length, dm.Length];77 77 double[,] b = new double[dm.Length, dm.Length]; 78 double[] ai = new double[dm.Length]; 79 double[] aj = new double[dm.Length]; 80 double aavg = 0.0; 81 double[] q = new double[dm.Length]; 82 double[,] v = new double[dm.Length, dm.Length]; 78 double[] q; //eigenvalues 79 double[,] v; //eigenvectors 83 80 81 double[][] x = SquareMatrix(dm); 82 CenterMatrix(x); 83 ChangeSignAndHalve(x); 84 84 85 //TODO: optimize memory consumption 85 86 for (int i = 0; i < dm.Length; i++) { 86 87 for (int j = 0; j < dm.Length; j++) { 87 a[i, j] = -0.5 * Math.Pow(dm[i][j], 2); 88 aavg += a[i, j]; 88 b[i, j] = x[i][j]; 89 89 } 90 90 } 91 91 92 aavg /= dm.Length * dm.Length; 92 bool res = alglib.smatrixevd(b, dm.Length, 1, true, out q, out v); 93 if (!res) throw new Exception("Eigenvalue computation did not converge!"); 93 94 94 for (int i = 0; i < dm.Length; i++) { 95 ai[i] = dm[i].Average(); 96 } 97 98 for (int i = 0; i < dm.Length; i++) { 95 //TODO: this should also work without allocating memory for ev and evec 96 double[] ev = new double[k]; 97 double[][] evec = new double[dm.Length][]; 98 AllocArray(evec, k); 99 Array.Copy(q, q.Length - k, ev, 0, k); 100 for (int i = 0; i < k; i++) { 99 101 for (int j = 0; j < dm.Length; j++) { 100 aj[i] += dm[j][i]; 101 } 102 aj[i] /= dm.Length; 103 } 104 105 for (int i = 0; i < dm.Length; i++) { 106 for (int j = 0; j < dm.Length; j++) { 107 b[i, j] = a[i, j] - ai[i] - aj[j] + aavg; 102 evec[j][i] = v[j, i + (q.Length - k)]; 108 103 } 109 104 } 110 105 111 //TODO: check b for negative values and make it positiv 112 113 double[] tau; 114 double[,] r; 115 alglib.rmatrixqr(ref b, dm.Length, dm.Length, out tau); 116 alglib.rmatrixqrunpackr(b, dm.Length, dm.Length, out r); 117 118 bool res = alglib.smatrixevd(r, dm.Length, 1, true, out q, out v); 119 if (!res) throw new Exception("Eigenvalue computation did not converge!"); 120 121 int zeroCnt = q.Count(x => x.IsAlmost(0) || x < 0.0); 122 for (int i = 0; i < dm.Length; i++) { 123 points[i] = new double[dm.Length - zeroCnt]; 106 double k1 = SumIfLZero(ev); 107 if (k1 < k) { 108 throw new Exception("Zero-eigenvalues detected. This leads to a degenerate point set. Use constants. "); 109 //TODO: handling of this case; implement adding of constants 124 110 } 125 111 126 int pi = 0; 127 for (int i = 0; i < dm.Length; i++) { 128 if (!q[i].IsAlmost(0.0) && q[i] > 0.0) { 129 for (int j = 0; j < dm.Length; j++) { 130 points[j][pi] = Math.Sqrt(q[i]) * v[j, i]; 131 } 132 pi++; 112 AllocArray(points, k); 113 for (int i = 0; i < k; i++) { 114 for (int j = 0; j < dm.Length; j++) { 115 points[j][i] = Math.Sqrt(ev[i]) * evec[j][i]; 133 116 } 134 117 } 118 return points; 119 } 135 120 136 return points; 121 //TODO: refactor the following methods into something sane 122 private static double[][] SquareMatrix(double[][] a) { 123 int n = a.Length; 124 double[][] newA = new double[a.Length][]; 125 126 for (int i = 0; i < n; i++) { 127 newA[i] = new double[a.Length]; 128 for (int j = 0; j < n; j++) { 129 newA[i][j] = Math.Pow(a[i][j], 2.0); 130 } 131 } 132 return newA; 133 } 134 135 //based on R's DoubleCentre 136 private static void CenterMatrix(double[][] a) { 137 int n = a.Length; 138 139 //reduce lines by line avg 140 for (int i = 0; i < n; i++) { 141 double sum = 0; 142 for (int j = 0; j < n; j++) sum += a[i][j]; 143 sum /= n; 144 for (int j = 0; j < n; j++) a[i][j] -= sum; 145 } 146 147 //reduce cols by col avg 148 for (int j = 0; j < n; j++) { 149 double sum = 0; 150 for (int i = 0; i < n; i++) sum += a[i][j]; 151 sum /= n; 152 for (int i = 0; i < n; i++) a[i][j] -= sum; 153 } 154 } 155 156 private static void ChangeSignAndHalve(double[][] a) { 157 int n = a.Length; 158 159 for (int i = 0; i < n; i++) { 160 for (int j = 0; j < n; j++) { 161 a[i][j] = (-1.0 * a[i][j]) / 2; 162 } 163 } 164 } 165 166 private static double SumIfLZero(double[] a) { 167 return a.Where(x => x > 0.0 && !x.IsAlmost(0.0)).Sum(); 168 } 169 170 private static void AllocArray(double[][] arr, int size) { 171 for (int i = 0; i < arr.Length; i++) { 172 arr[i] = new double[size]; 173 } 137 174 } 138 175
Note: See TracChangeset
for help on using the changeset viewer.