javascript實現數據結構:稀疏矩陣的十字鏈表存儲表示

當矩陣的非零個數和位置在操做過程當中變化大時,就不宜採用順序存儲結構來表示三元組的線性表。例如,在做「將矩陣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);
相關文章
相關標籤/搜索