Сортировка вложенного объекта по свойству не работает - PullRequest
1 голос
/ 11 апреля 2019

Я пытаюсь отсортировать объект players по свойству average экземпляров player внутри него, но не могу получить доступ, чтобы он работал.

Вот как выглядит объект Players, когда я вхожу в него:

[ Player { player_id: 1, scores: [ 50 ], total: 50, average: 50 },
  Player { player_id: 2, scores: [ 80, 70, 60 ], total: 210, average: 80 },
  Player { player_id: 3, scores: [ 90, 85 ], total: 175, average: 90 } ]

Но моя функция сортировки, похоже, не работает - думаю, мне нужно добавить куда-нибудь player, чтобы получить доступ к вложенным объектам.

Вот этот код функции:

/* class definition excluded */

    top = (num_players) => 
      Object.values(this.players)
        .sort((a, b) => (a.average - b.average))
        .slice(0, num_players);
};

Тест, написанный для этой функции, возвращает следующее:

Top 3 [[object Object],[object Object],[object Object]] should equal [3, 2, 1]:

Для контекста, вот весь сценарий с тестовыми примерами, на случай, если вы захотите попробовать его:

class Player {
  constructor(player_id, score) {
    this.player_id = player_id;
    this.scores = [score];
    this.total = score;
    this.average = this.averageScore();
  }

  addScore(score) {
    this.total += score;
    this.scores.push(score);
    return score;
  }

  averageScore() {
    return this.scores.length ? this.total / this.scores.length : 0;
  }

  resetScore() {
    this.scores = [];
    this.score = 0;
  }

};

class LeaderBoard {

  constructor() {
    this.players = {};
  }

  add_score(player_id, score) {
    if (!this.players[player_id]) {
      this.players[player_id] = new Player(player_id, score);
    } else {
      this.players[player_id].addScore(score);
    }
    var average = this.players[player_id].averageScore();
    return average.toFixed(1);
  }

  top = (num_players) =>
    // console.log(Object.values(this.players);
    Object.values(this.players)
    .sort((a, b) => (a.average - b.average))
    .slice(0, num_players);

  reset = (player_id) => {
    this.players[player_id].resetScore();
  }
};

// Test code here

function array_equals(a, b) {
  if (a === b) return true;
  if (a == null || b == null) return false;
  if (a.length != b.length) return false;
  for (var i = 0; i < a.length; ++i) {
    if (a[i] !== b[i]) return false;
  }
  return true;
}

var leader_board = new LeaderBoard()

leader_board.add_score(1, 50)
console.log(leader_board.add_score(2, 80) == 80)
console.log(leader_board.add_score(2, 70) == 75)
console.log(leader_board.add_score(2, 60) == 70)
console.log('Add score should return the average. test with 1 score')
console.log(leader_board.add_score(3, 90) == 90)
console.log('Add score should return the average. test with 2 scores')
console.log(leader_board.add_score(3, 85) == 87.5)
console.log('Top 3 [' + leader_board.top(3) + '] should equal [3, 2, 1]:')
console.log(array_equals(leader_board.top(3), [3, 2, 1]))
console.log('Top 2 [' + leader_board.top(2) + '] should equal [3, 2]:')
console.log(array_equals(leader_board.top(2), [3, 2]))
leader_board.reset(3)
console.log('After reset top 3 [' + leader_board.top(3) + '] should equal [2, 1, 3]')
console.log(array_equals(leader_board.top(3), [2, 1, 3]))

Ожидается, что top сможет отсортировать объект players по их average, а затем вернуть их в массив из 0 to num_players, но я не могу выполнить сортировку успешно.

Пока что мне не повезло - как отсортировать по вложенному свойству?

1 Ответ

3 голосов
/ 11 апреля 2019
  1. Вы сравниваете Player объекты со свойством самого объекта (player_id). Правильное утверждение assert будет выглядеть примерно так:
console.log(array_equals(leader_board.top(3).map(p => p.player_id), [3, 2, 1]))
  1. Вы сортируете функцию в неправильном направлении, просто переключите ее на:
    sort((a, b) => (b.average - a.average))
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...