Как удалить дубликаты и подсчитать общее количество записей в массиве по идентификатору - PullRequest
0 голосов
/ 03 июля 2018

Упражнение выглядит следующим образом: Объект, в котором ключами являются названия банков и значения количества клиентов, которые имеют счета только в этом банке.

с этим массивом:

const accounts = [ 
   { clientId: 6, bankId: 1, balance: 15000 }, 
   { clientId: 1, bankId: 3, balance: 18000 },
   { clientId: 5, bankId: 3, balance: 135000 },
   { clientId: 2, bankId: 2, balance: 5600 },
   { clientId: 3, bankId: 1, balance: 23000 },
   { clientId: 5, bankId: 2, balance: 15000 },
   { clientId: 3, bankId: 3, balance: 45900 },
   { clientId: 2, bankId: 3, balance: 19000 },
   { clientId: 4, bankId: 3, balance: 51000 },
   { clientId: 5, bankId: 1, balance: 89000 },
   { clientId: 1, bankId: 2, balance: 1600 },
   { clientId: 5, bankId: 3, balance: 37500 },
   { clientId: 6, bankId: 1, balance: 19200 },
   { clientId: 2, bankId: 3, balance: 10000 },
   { clientId: 3, bankId: 2, balance: 5400 },
   { clientId: 3, bankId: 1, balance: 9000 },
   { clientId: 4, bankId: 3, balance: 13500 },
   { clientId: 2, bankId: 1, balance: 38200 },
   { clientId: 5, bankId: 2, balance: 17000 },
   { clientId: 1, bankId: 3, balance: 1000 },
   { clientId: 5, bankId: 2, balance: 600 },
   { clientId: 6, bankId: 1, balance: 16200 },
   { clientId: 2, bankId: 2, balance: 10000 }
];

У меня есть это сейчас:

function banksFidelity() {
  var map = accounts.reduce(function(map, account) {
    var bankByid = account.bankId;
    var clientByid = account.clientId;

    map[bankByid] = map[clientByid] || 0 + clientByid

    return map

  }, {})

  return map

}

Текущий результат:

{1: 6, 2: 5, 3: 5}

Желаемый результат:

{1: 4, 2: 4, 3: 4}

Ответы [ 3 ]

0 голосов
/ 03 июля 2018

Вы можете уменьшить массив, чтобы удалить все дубликаты, а затем уменьшить этот массив, чтобы сформировать конечный объект.

const accounts = [ 
   { clientId: 6, bankId: 1, balance: 15000 }, 
   { clientId: 1, bankId: 3, balance: 18000 },
   { clientId: 5, bankId: 3, balance: 135000 },
   { clientId: 2, bankId: 2, balance: 5600 },
   { clientId: 3, bankId: 1, balance: 23000 },
   { clientId: 5, bankId: 2, balance: 15000 },
   { clientId: 3, bankId: 3, balance: 45900 },
   { clientId: 2, bankId: 3, balance: 19000 },
   { clientId: 4, bankId: 3, balance: 51000 },
   { clientId: 5, bankId: 1, balance: 89000 },
   { clientId: 1, bankId: 2, balance: 1600 },
   { clientId: 5, bankId: 3, balance: 37500 },
   { clientId: 6, bankId: 1, balance: 19200 },
   { clientId: 2, bankId: 3, balance: 10000 },
   { clientId: 3, bankId: 2, balance: 5400 },
   { clientId: 3, bankId: 1, balance: 9000 },
   { clientId: 4, bankId: 3, balance: 13500 },
   { clientId: 2, bankId: 1, balance: 38200 },
   { clientId: 5, bankId: 2, balance: 17000 },
   { clientId: 1, bankId: 3, balance: 1000 },
   { clientId: 5, bankId: 2, balance: 600 },
   { clientId: 6, bankId: 1, balance: 16200 },
   { clientId: 2, bankId: 2, balance: 10000 }
]

let result = accounts
  // Remove duplicates
  .reduce((arr, itm) => {
    let containsClient = !!arr.find(i => i.clientId == itm.clientId && i.bankId == itm.bankId)
    return !containsClient ? arr.concat(itm) : arr
  }, [])
  // Get the counts after duplicates are removed
  .reduce((obj, itm) => {
    if (obj[itm.bankId]) obj[itm.bankId]++
    else obj[itm.bankId] = 1
    return obj
  }, {})

console.log(result)
0 голосов
/ 03 июля 2018

вариант: используя ссылки, поэтому, когда мы обновляем клиента один раз, мы можем извлечь информацию из того места, где он был добавлен / ссылка скопирована

 const accounts = [ 
   { clientId: 6, bankId: 1, balance: 15000 }, 
   { clientId: 1, bankId: 3, balance: 18000 },
   { clientId: 5, bankId: 3, balance: 135000 },
   { clientId: 2, bankId: 2, balance: 5600 },
   { clientId: 3, bankId: 1, balance: 23000 },
   { clientId: 5, bankId: 2, balance: 15000 },
   { clientId: 3, bankId: 3, balance: 45900 },
   { clientId: 2, bankId: 3, balance: 19000 },
   { clientId: 4, bankId: 3, balance: 51000 },
   { clientId: 5, bankId: 1, balance: 89000 },
   { clientId: 1, bankId: 2, balance: 1600 },
   { clientId: 5, bankId: 3, balance: 37500 },
   { clientId: 6, bankId: 1, balance: 19200 },
   { clientId: 2, bankId: 3, balance: 10000 },
   { clientId: 3, bankId: 2, balance: 5400 },
   { clientId: 3, bankId: 1, balance: 9000 },
   { clientId: 4, bankId: 3, balance: 13500 },
   { clientId: 2, bankId: 1, balance: 38200 },
   { clientId: 5, bankId: 2, balance: 17000 },
   { clientId: 1, bankId: 3, balance: 1000 },
   { clientId: 5, bankId: 2, balance: 600 },
   { clientId: 6, bankId: 1, balance: 16200 },
   { clientId: 2, bankId: 2, balance: 10000 }
  ]

var clients = {}
var dupes = {}
var banks = {}
accounts.forEach(a=>{
    if(dupes[a.clientId+'_'+a.bankId]){
        return;
    }
    banks[a.bankId] = banks[a.bankId] || new Set;
    if(clients[a.clientId]){
        //this client has already a bank
        //different than the current one
        clients[a.clientId].multi = true;
    }else{
        clients[a.clientId]={}
    }
    banks[a.bankId].add(a.clientId)
    dupes[a.clientId+'_'+a.bankId] = 1;
}, {})
var res = Object.keys(banks).reduce((acc,bId)=>{
    acc[bId] = [...banks[bId]].filter(x=>!x.multi).length
    return acc;
},{})
console.log('res ', res)
0 голосов
/ 03 июля 2018

Вы можете перебрать объект массива счетов с помощью метода .foreach и заполнить другой объект по мере необходимости, проверив, есть ли у клиента счет в этом банке. У вас остается объект (clients), связывающий каждого клиента с массивом банков, которые он использует:

{
  "1": [3, 2],
  "2": [2, 3, 1],
  "3": [1, 3, 2],
  "4": [3],
  "5": [3, 2, 1],
  "6": [1]
}

Используйте другой цикл .foreach, чтобы заполнить другой объект (exclusiveClients), который будет связывать банки с количеством клиентов, которые используют этот банк только для получения желаемого результата:

{
  "1": 1,
  "3": 1
}

Вот код:

const accounts=[{clientId:6,bankId:1,balance:15000},{clientId:1,bankId:3,balance:18000},{clientId:5,bankId:3,balance:135000},{clientId:2,bankId:2,balance:5600},{clientId:3,bankId:1,balance:23000},{clientId:5,bankId:2,balance:15000},{clientId:3,bankId:3,balance:45900},{clientId:2,bankId:3,balance:19000},{clientId:4,bankId:3,balance:51000},{clientId:5,bankId:1,balance:89000},{clientId:1,bankId:2,balance:1600},{clientId:5,bankId:3,balance:37500},{clientId:6,bankId:1,balance:19200},{clientId:2,bankId:3,balance:10000},{clientId:3,bankId:2,balance:5400},{clientId:3,bankId:1,balance:9000},{clientId:4,bankId:3,balance:13500},{clientId:2,bankId:1,balance:38200},{clientId:5,bankId:2,balance:17000},{clientId:1,bankId:3,balance:1000},{clientId:5,bankId:2,balance:600},{clientId:6,bankId:1,balance:16200},{clientId:2,bankId:2,balance:10000}]

let clients = {};

accounts.forEach(({bankId, clientId}) => {

  if(clients[clientId] && !clients[clientId].includes(bankId)) {
  
    clients[clientId].push(bankId);
  
  } else if(!clients[clientId]) {
  
    clients[clientId] = [bankId];
  
  }
  
});


let exclusiveClients = {};

Object.entries(clients).forEach(([client, banks]) => {

  if(banks.length == 1) {
  
    const bank = banks[0];
    exclusiveClients[bank] = (exclusiveClients[bank] + 1) || 1;
    
    
  }
  
});

console.log(exclusiveClients)
...