web-dev-qa-db-ja.com

javascriptでネストされた配列の最大/最小を見つける方法は?

次のようなネストされた配列の最大値を見つけたい:

a = [[1,2],[20,3]]
d3.max(d3.max(a)) // 20

しかし、私の配列には破棄したいテキストフィールドが含まれています:

a = [["yz",1,2],["xy",20,3]]
d3.max(a) // 20
30
nachocab

数値のネストされた配列(arrays = [[1, 2], [20, 3]])、ネスト d3.max

var max = d3.max(arrays, function(array) {
  return d3.max(array);
});

または同等に、 array.map を使用します。

var max = d3.max(arrays.map(function(array) {
  return d3.max(array);
}));

文字列値を無視する場合は、 array.filter を使用して文字列を無視できます。

var max = d3.max(arrays, function(array) {
  return d3.max(array.filter(function(value) {
    return typeof value === "number";
  }));
});

あるいは、文字列が常に最初の位置にあることがわかっている場合は、 array.slice を使用できます。これはもう少し効率的です。

var max = d3.max(arrays, function(array) {
  return d3.max(array.slice(1));
});

さらに別のオプションは、数値ではない値に対してNaNを返すアクセサー関数を使用することです。これにより、d3.maxはこれらの値を無視します。便利なことに、JavaScriptの組み込みNumber関数はまさにこれを行うため、次のように言えます。

var max = d3.max(arrays, function(array) {
  return d3.max(array, Number);
});
74
mbostock

これを使って:

function arrmax(arrs) {
    var toplevel = [];

    var f = function(v) {
        return !isNaN(v);
    };

    for (var i = 0, l = arrs.length; i<l; i++) {
        toplevel.Push(Math.max.apply(window, arrs[i].filter(f)));
    }
    return Math.max.apply(window, toplevel);
}

またはそれ以上:

function arrmax(arrs) {
    if (!arrs || !arrs.length) return undefined;
    var max = Math.max.apply(window, arrs[0]), m,
        f = function(v){ return !isNaN(v); };
    for (var i = 1, l = arrs.length; i<l; i++) {
        if ((m = Math.max.apply(window, arrs[i].filter(f)))>max) max=m;
    }
    return max;
}

Array.filterメソッドの詳細については [〜#〜] mdn [〜#〜] をご覧ください。

4
kbec

テストする列を正確に指定した場合、次を使用できます。

var columns = ["ColumnA", "ColumnB", "ColumnC"];

var max = selectedMax(columns,dataset);
var min = selectedMin(columns,dataset)

function selectedMax(columns, dataset) {
    var max;
    columns.forEach(function(element, index, array) {
        var tmpmax = d3.max(dataset, function(d) {
            return +d[element];
        });       
        max = (tmpmax > max || max === undefined) ? tmpmax : max;
    });
    return max;
}

function selectedMin(columns, dataset) {
    var min;
    columns.forEach(function(element, index, array) {
        var tmpmin = d3.min(dataset, function(d) {
            return +d[element];
        });
        min = (tmpmin < min || min === undefined) ? tmpmin : min;
    });  
return min;
}
2
Rick

配列をフラット化し、各メンバーに関数を適用できます

Array.prototype.flatten= function(fun){
    if(typeof fun!= 'function') fun= '';
    var A= [], L= this.length, itm;
    for(var i= 0; i<L; i++){
        itm= this[i];
        if(itm!= undefined){
            if(!itm.flatten){
                if(fun) itm= fun(itm);
                if(itm) A.Push(itm);
            }
            else A= A.concat(itm.flatten(fun));
        }
    }
    return A;
}

var a= [["yz", 1, 2], ["xy", 20, 3]], max=-Infinity;

var max=Math.max.apply(a, a.flatten(Number));
2
kennebec

d3.arrayd3.merge 配列の配列を平坦化します。

と相まって - d3.max およびアクセサとしてのjavascriptのNumber

var max = d3.max(d3.merge(arrays), Number);

例えば:

var input = [["yz", 1, 2], ["xy", 20, 3]];

var max = d3.max(d3.merge(input), Number);

console.log(max);
<script src="https://d3js.org/d3-array.v2.min.js"></script>
0
Xavier Guihot

それは残酷なハックですが、_d3.max_の ソースコード を見ると、そのコードをコピーすることで最初の要素を破棄する_d3.max1_を定義することができますが、 _i=-1_と_i=0_。そのリンクのコードはここに抜粋されています。私は通常のd3.jsユーザーではありませんが、ライブラリについて知っていることから、応答できるように、この関数のようにバージョンに_f.call_ケースがあることを確認する必要があります。更新を正しく実行します。

_d3.max = function(array, f) {
  var i = -1,
      n = array.length,
      a,
      b;
  if (arguments.length === 1) {
    while (++i < n && ((a = array[i]) == null || a != a)) a = undefined;
    while (++i < n) if ((b = array[i]) != null && b > a) a = b;
  } else {
    while (++i < n && ((a = f.call(array, array[i], i)) == null || a != a)) a = undefined;
    while (++i < n) if ((b = f.call(array, array[i], i)) != null && b > a) a = b;
  }
  return a;
};
_

すると、それはd3.max(d3.max1(a))になります。

0
btown