當矩陣的非零個數和位置在操做過程當中變化大時,就不宜採用順序存儲結構來表示三元組的線性表。例如,在做「將矩陣B加到矩陣A上」的操做時,因爲非零元的插入或刪除將會引發A.data中元素的移動。爲此,對這種類型的矩陣,採用鏈式存儲結構表示三元組的線性表更爲恰當。數組
在鏈表中,每一個非陵園可用一個含5個域的結點表示,其中i,j和e這3個域分別表示該非零元所在的行,列和非零元的值,向右域right用以連接同一行中下一個非零元,向下域down用以連接同一列中下一個非零元。同一行的非零元經過right域連接成一個線性表,同一列中的非零元一般down域連接成一個線性鏈表,每個非零元既是某個行鏈表中的一個結點,又是某個列鏈表中的一個結點,整個矩陣構成了一個十字交叉的鏈表。app
可用兩個分別存儲行鏈表的頭指針和列鏈表的頭指針的一維數組來表示。this
代碼:spa
1 /** 2 * 十字鏈表 3 * 4 * 當矩陣的非零個數和位置在操做過程當中變化大時,就不宜採用順序存儲結構來表示三元組的線性表。例如,在做「將矩陣B加到矩陣A上」的操做時,因爲非零元的插入或刪除將會引發A.data中元素的移動。爲此,對這種類型的矩陣,採用鏈式存儲結構表示三元組的線性表更爲恰當。 5 * 6 * 在鏈表中,每一個非陵園可用一個含5個域的結點表示,其中i,j和e這3個域分別表示該非零元所在的行,列和非零元的值,向右域right用以連接同一行中下一個非零元,向下域down用以連接同一列中下一個非零元。同一行的非零元經過right域連接成一個線性表,同一列中的非零元一般down域連接成一個線性鏈表,每個非零元既是某個行鏈表中的一個結點,又是某個列鏈表中的一個結點,整個矩陣構成了一個十字交叉的鏈表。 7 * 8 * 可用兩個分別存儲行鏈表的頭指針和列鏈表的頭指針的一維數組來表示。 9 */ 10 11 // 稀疏矩陣的十字鏈表存儲表示 12 13 function OLNode(i, j, e) { 14 // 該非零元的行和列下標 15 this.i = i || 0; 16 this.j = j || 0; 17 this.e = e; 18 // 該非零元所在行表和列表的後繼鏈域 19 this.right = null; // type: OLNode 20 this.down = null; // type: OLNode 21 } 22 23 function CrossList() { 24 // 行和列鏈表頭指針向量基址由CreateSMatrix分配 25 this.rhead = []; 26 this.chead = []; 27 // 稀疏矩陣的行數,列數 28 this.mu = 0; 29 this.nu = 0; 30 this.tu = 0; 31 } 32 /** 33 * 矩陣初始化 34 * @param m 35 * @param n 36 * @param t 37 * @param {Array} list 二維數組,每行的元素分別是[i, j, e] 38 */ 39 CrossList.prototype.createSMatrix = function (m, n, t, list) { 40 this.mu = m; 41 this.nu = n; 42 this.tu = t; 43 44 for (var row = 0; row < list.length; row++) { 45 var p = {}; 46 OLNode.apply(p, list[row]); 47 var i = list[row][0]; 48 var j = list[row][1]; 49 var q; 50 51 if (this.rhead[i] == null || this.rhead[i].j > j) { 52 p.right = this.rhead[i]; 53 this.rhead[i] = p; 54 } else { 55 // 查詢在行表中的插入位置 56 for (q = this.rhead[i]; q.right && q.right.j < j; q = q.right); 57 p.right = q.right; 58 q.right = p; 59 } 60 61 if (this.chead[j] == null || this.chead[j].i > i) { 62 p.down = this.chead[j]; 63 this.chead[j] = p; 64 } else { 65 for (q = this.chead[j]; q.down && q.down.i < i; q = q.down); 66 p.down = q.down; 67 q.down = p; 68 } 69 } 70 }; 71 72 // 矩陣相加 73 CrossList.prototype.addMatrix = function (crossList) { 74 var hl = []; 75 //hl初始化 76 for (var j = 0; j <= this.nu; j++) 77 hl[j] = this.chead[j]; 78 79 for (var i = 0; i <= this.mu; i++) { 80 //pa和pb指向每一行的第一個非0元結點,直至最後一行 81 var pa = this.rhead[i]; 82 var pb = crossList.rhead[i]; 83 var pre = null; 84 85 //處理B的一行,直至本行中無非0元素的結點 86 while (pb) { 87 var p, q; 88 // 新插入一個結點到pa的左側 89 if (!pa || pa.j > pb.j) { 90 p = new OLNode(pb.i, pb.j, pb.e); 91 92 //行表的指針變化 93 if (!pre) this.rhead[p.i] = p; 94 else pre.right = p; 95 96 p.right = pa; 97 pre = p; 98 99 //列表的指針變化 100 if (hl[p.j]) { 101 // 從hl[p.j]開始找到新結點在同一列中的前驅結點,並讓hl[p.j]指向它 102 for (q = hl[p.j]; q && q.i < p.i;q = q.down) 103 hl[p.j] = q; 104 } 105 106 //在列表中插入新結點,根據行數判斷插入前面仍是後面 107 if (!this.chead[p.j] || this.chead[p.j].i > p.i) { 108 p.down = this.chead[p.j]; 109 this.chead[p.j] = p; 110 } else { 111 p.down = hl[p.j].down; 112 hl[p.j].down = p; 113 } 114 115 hl[p.j] = p; 116 pb = pb.right; 117 } else if (pa.j < pb.j) { 118 pre = pa; 119 pa = pa.right; 120 } else { 121 //當pa.j === pb.j時,將B中當前結點的值加到A中當前結點上 122 pa.e += pb.e; 123 124 //當pa.e === 0時,刪除該結點 125 if (pa.e === 0) { 126 // 若無前驅結點,將第一個非0元結點置爲當前結點的後繼結點, 127 // 不然前驅結點的後繼結點爲當前結點的後繼結點 128 if (!pre) this.rhead[pa.i] = pa.right; 129 else pre.right = pa.right; 130 131 p = pa; 132 pa = pa.right; 133 134 //列表的指針變化 135 if (hl[p.j]) { 136 //從hl[p.j]開始找到新結點在同一列中的前驅結點,並讓hl[p.j]指向它 137 for (q = hl[p.j]; q && q.i < p.i; q = q.down) 138 hl[p.j] = q; 139 } 140 141 if (this.chead[p.j] == p) 142 this.chead[p.j] = hl[p.j] = p.down; 143 else 144 hl[p.j].down = p.down; 145 } 146 147 pb = pb.right; 148 } 149 } 150 } 151 }; 152 153 var lists = [ 154 [1, 4, 5], 155 [2, 2, -1], 156 [1, 1, 3], 157 [3, 1, 2] 158 ]; 159 var a = new CrossList(); 160 a.createSMatrix(4, 4, 4, lists); 161 console.log(a); 162 163 var lists2 = [ 164 [1, 4, -5], 165 [2, 3, 1], 166 [1, 1, 3], 167 [3, 2, 2] 168 ]; 169 var b = new CrossList(); 170 b.createSMatrix(4, 4, 4, lists2); 171 console.log(b); 172 173 a.addMatrix(b); 174 console.log(a);