admin管理员组

文章数量:1341401

I have an array of objects in javascript (D3) and I need to remove every object of which a certain attribute is present in another array of objects attribute,
i.e. a left outer join
(source: tazindeed.co.uk)

I managed to do it myself with 2 loops but it's quite slow.
And I don't know how to make it faster.

    for (var i = 0; i < data1.length; i++) {
        for (var j = 0; j < data2.length; j++) {
            if (data2[j].attr3 == data1[i].attr4) {
                data2.splice(j,1);
            }
        }
    }

data1.length~2k and data2.length~10k

I know this has approximately been asked here but it's been almost 2 years and the solutions use external libraries.
I'm just curious to learn if there is a better method with javascript (or jQuery or D3, which I already use)

Thank you for your help !

I have an array of objects in javascript (D3) and I need to remove every object of which a certain attribute is present in another array of objects attribute,
i.e. a left outer join
(source: tazindeed.co.uk)

I managed to do it myself with 2 loops but it's quite slow.
And I don't know how to make it faster.

    for (var i = 0; i < data1.length; i++) {
        for (var j = 0; j < data2.length; j++) {
            if (data2[j].attr3 == data1[i].attr4) {
                data2.splice(j,1);
            }
        }
    }

data1.length~2k and data2.length~10k

I know this has approximately been asked here but it's been almost 2 years and the solutions use external libraries.
I'm just curious to learn if there is a better method with javascript (or jQuery or D3, which I already use)

Thank you for your help !

Share Improve this question edited Sep 26, 2019 at 3:31 Glorfindel 22.7k13 gold badges89 silver badges119 bronze badges asked Mar 2, 2015 at 12:53 stallingOnestallingOne 4,0263 gold badges49 silver badges69 bronze badges
Add a ment  | 

4 Answers 4

Reset to default 4

You need a fast lookup for the values that exist in data1, so make a map using an object:

var map = {};
for (var i = 0; i < data1.length; i++) {
  map[data1[i].attr4] = 1;
}

Then you can loop throught the items in data2 and filter them:

var result = [];
for (i = 0; i < data2.length; i++) {
  if (!(data2[i].attr3 in map)) {
    result.push(data2[i]);
  }
}

Maybe not faster but more readable

const left = ['1', '2', '7']
const right = ['1', '3', '5', '9']

const result = left.filter((x) => !right.includes(x))

You could use Array.filter (see MDN) or Array.map for that I suppose:

var array1 = [1,2,3,4,5,6,7,8,9],
    array2 = [3,4,5,6],
    fltr   = array1.filter( function(v) {return this.indexOf(v) < 0;}, array2),
    map    = array1.map( 
              function(v) {
                 var exists = this.indexOf(v);
                 return [v, (this[exists] || 'null')].join(', ');}, 
              array2),
    result = document.querySelector('#result');
fltr.unshift('<u>array1</u>');
map.unshift('<u>array1, array2</u>');
result.innerHTML = ['<b>filtered</b>\n',
                     fltr.join('\n'),
                    '\n\n<b>mapped</b>\n',
                     map.join('\n')].join('');
  
<pre id="result"></pre>

If the data is unique, you can now FINALLY use Sets to left outer join (using difference) I think as of a week ago at least using Chrome.

example - convert your arrays to sets:

const frontEndLanguages = new Set(["JavaScript", "HTML", "CSS"]);

const backEndLanguages = new Set(["Python", "Java", "JavaScript"]);

const onlyFrontEnd = frontEndLanguages.difference(backEndLanguages);
// => Set {"HTML", "CSS"} 

const onlyBackEnd = backEndLanguages.difference(frontEndLanguages);
// => Set {"Python", "Java"}

本文标签: jquerymake a left outer join in javascriptStack Overflow