123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208 |
- // Hirschberg's algorithm
- // http://en.wikipedia.org/wiki/Hirschberg%27s_algorithm
- /**
- * @module zrender/core/arrayDiff
- * @author Yi Shen
- */
- function defaultCompareFunc(a, b) {
- return a === b;
- }
- function createItem(cmd, idx, idx1) {
- var res = {
- // cmd explanation
- // '=': not change
- // '^': replace with a new item in second array. Unused temporary
- // '+': add a new item of second array
- // '-': del item in first array
- cmd: cmd,
- // Value index, use index in the first array
- // Except '+'. Adding a new item needs value in the second array
- idx: idx
- }; // Replace need to know both two indices
- // if (cmd === '^') {
- // res.idx1 = idx1;
- // }
- if (cmd === '=') {
- res.idx1 = idx1;
- }
- return res;
- }
- function append(out, cmd, idx, idx1) {
- out.push(createItem(cmd, idx, idx1));
- }
- var abs = Math.abs; // Needleman-Wunsch score
- function score(arr0, arr1, i0, i1, j0, j1, equal, memo) {
- var last;
- var invM = i0 > i1;
- var invN = j0 > j1;
- var m = abs(i1 - i0);
- var n = abs(j1 - j0);
- var i;
- var j;
- for (i = 0; i <= m; i++) {
- for (j = 0; j <= n; j++) {
- if (i === 0) {
- memo[j] = j;
- } else if (j === 0) {
- last = memo[j];
- memo[j] = i;
- } else {
- // memo[i-1][j-1] + same(arr0[i-1], arr1[j-1]) ? 0 : 1
- // Retained or replace
- var val0 = arr0[invM ? i0 - i : i - 1 + i0];
- var val1 = arr1[invN ? j0 - j : j - 1 + j0]; // Because replace is add after remove actually
- // It has a higher score than removing or adding
- // TODO custom score function
- var score0 = last + (equal(val0, val1) ? 0 : 2); // memo[i-1][j] + 1
- // Remove arr0[i-1]
- var score1 = memo[j] + 1; // memo[i][j-1] + 1
- // Add arr1[j-1]
- var score2 = memo[j - 1] + 1;
- last = memo[j];
- memo[j] = score0 < score1 ? score0 : score1;
- score2 < memo[j] && (memo[j] = score2); // Math min of three parameters seems slow
- // memo[j] = Math.min(score0, score1, score2);
- }
- }
- }
- return memo;
- }
- function hirschberg(arr0, arr1, i0, i1, j0, j1, equal, score0, score1) {
- var out = [];
- var len0 = i1 - i0;
- var len1 = j1 - j0;
- var i;
- var j;
- if (!len0) {
- for (j = 0; j < len1; j++) {
- append(out, '+', j + j0);
- }
- } else if (!len1) {
- for (i = 0; i < len0; i++) {
- append(out, '-', i + i0);
- }
- } else if (len0 === 1) {
- var a = arr0[i0];
- var matched = false;
- for (j = 0; j < len1; j++) {
- if (equal(a, arr1[j + j0]) && !matched) {
- matched = true; // Equal and update use the index in first array
- append(out, '=', i0, j + j0);
- } else {
- // if (j === len1 - 1 && ! matched) {
- // append(out, '^', i0, j + j0);
- // }
- // else {
- append(out, '+', j + j0); // }
- }
- }
- if (!matched) {
- append(out, '-', i0);
- }
- } else if (len1 === 1) {
- var b = arr1[j0];
- var matched = false;
- for (i = 0; i < len0; i++) {
- if (equal(b, arr0[i + i0]) && !matched) {
- matched = true;
- append(out, '=', i + i0, j0);
- } else {
- // if (i === len0 - 1 && ! matched) {
- // append(out, '^', i + i0, j0);
- // }
- // else {
- append(out, '-', i + i0); // }
- }
- }
- if (!matched) {
- append(out, '+', j0);
- }
- } else {
- var imid = (len0 / 2 | 0) + i0;
- score(arr0, arr1, i0, imid, j0, j1, equal, score0);
- score(arr0, arr1, i1, imid + 1, j1, j0, equal, score1);
- var min = Infinity;
- var jmid = 0;
- var sum;
- for (j = 0; j <= len1; j++) {
- sum = score0[j] + score1[len1 - j];
- if (sum < min) {
- min = sum;
- jmid = j;
- }
- }
- jmid += j0;
- out = hirschberg(arr0, arr1, i0, imid, j0, jmid, equal, score0, score1);
- var out1 = hirschberg(arr0, arr1, imid, i1, jmid, j1, equal, score0, score1); // Concat
- for (i = 0; i < out1.length; i++) {
- out.push(out1[i]);
- }
- }
- return out;
- }
- function arrayDiff(arr0, arr1, equal) {
- equal = equal || defaultCompareFunc; // Remove the common head and tail
- var i;
- var j;
- var len0 = arr0.length;
- var len1 = arr1.length;
- var lenMin = Math.min(len0, len1);
- var head = [];
- for (i = 0; i < lenMin; i++) {
- if (!equal(arr0[i], arr1[i])) {
- break;
- }
- append(head, '=', i, i);
- }
- for (j = 0; j < lenMin; j++) {
- if (!equal(arr0[len0 - j - 1], arr1[len1 - j - 1])) {
- break;
- }
- }
- if (len0 - j >= i || len1 - j >= i) {
- var middle = hirschberg(arr0, arr1, i, len0 - j, i, len1 - j, equal, [], []);
- for (i = 0; i < middle.length; i++) {
- head.push(middle[i]);
- }
- for (i = 0; i < j; i++) {
- append(head, '=', len0 - j + i, len1 - j + i);
- }
- }
- return head;
- }
- var _default = arrayDiff;
- module.exports = _default;
|