javascript - Pushing to an array inside an object - Stack Overflow

I just recently picked up JS and have been working on an exercise to sort people from an array of recor

I just recently picked up JS and have been working on an exercise to sort people from an array of records, ancestry, into the centuries they lived in and their ages. Here's a sample element from ancestry:

{
name:   "Carolus Haverbeke"
sex:    "m"
born:   1832
died:   1905
father: "Carel Haverbeke"
mother: "Maria van Brussel"
}

Here's my attempt:

ancestry.forEach(function(person) {
  var ages = {};
  var century = Math.ceil(person.died / 100);

  if (century in ages) 
    ages[century].push(person.died - person.born);
  else
    ages[century] = person.died - person.born;
});

This is the general format I'm trying to store in ages, which maps each century to an array of the people's ages:

{
16: [24, 51, 16]
17: [73, 22]
18: [54, 65, 28]
}

I'm really confused as this code only saves the first person in ancestry into ages. I thought it might be because I defined ages inside the forEach, but when I move var ages = {} outside I get this:

TypeError: ages[century].push is not a function (line 18)

Could someone please help explain what's going on and the code should be fixed?

I just recently picked up JS and have been working on an exercise to sort people from an array of records, ancestry, into the centuries they lived in and their ages. Here's a sample element from ancestry:

{
name:   "Carolus Haverbeke"
sex:    "m"
born:   1832
died:   1905
father: "Carel Haverbeke"
mother: "Maria van Brussel"
}

Here's my attempt:

ancestry.forEach(function(person) {
  var ages = {};
  var century = Math.ceil(person.died / 100);

  if (century in ages) 
    ages[century].push(person.died - person.born);
  else
    ages[century] = person.died - person.born;
});

This is the general format I'm trying to store in ages, which maps each century to an array of the people's ages:

{
16: [24, 51, 16]
17: [73, 22]
18: [54, 65, 28]
}

I'm really confused as this code only saves the first person in ancestry into ages. I thought it might be because I defined ages inside the forEach, but when I move var ages = {} outside I get this:

TypeError: ages[century].push is not a function (line 18)

Could someone please help explain what's going on and the code should be fixed?

Share Improve this question edited Apr 28, 2017 at 5:56 cafekaze asked Apr 28, 2017 at 5:30 cafekazecafekaze 3772 gold badges7 silver badges20 bronze badges 0
Add a ment  | 

4 Answers 4

Reset to default 6

tl;dr

var ancestry = [{ ... }, { ... }, ...],
    ages = {};

ancestry.forEach(function (person) {
    var century = Math.ceil(person.died / 100),
        age = person.died - person.born;

    !ages[century] && (ages[century] = []);

    ages[century].push(age);
});

console.log(ages);

We will discribe your code to understand problems:

Assume we have an ancestry Array (that means var ancestry = [{...}, {...}, ...] is in this form and each item look like this:

var ancestry = [{
    name: "Carolus Haverbeke",
    sex:  "m",
    born: 1832,
    died: 1905,
    father: "Carel Haverbeke",
    mother: "Maria van Brussel"
}, {
    name: "Carolus Haverbeke",
    sex:  "m",
    born: 1722,
    died: 1805,
    father: "Carel Haverbeke",
    mother: "Maria van Brussel"
}, {
    name: "Carolus Haverbeke",
    sex:  "m",
    born: 1666,
    died: 1705,
    father: "Carel Haverbeke",
    mother: "Maria van Brussel"
}, {
    name: "Carolus Haverbeke",
    sex:  "m",
    born: 1622,
    died: 1715,
    father: "Carel Haverbeke",
    mother: "Maria van Brussel"
}];

with different data.

To achieve what you want, follow this steps:

// Create your variable in the parent scope to allow variables to exist after the end of `forEach` method.
var ages = {};

// In the ancestry Array, we will loop on all items like `{ ... }`.
ancestry.forEach(function (person) {
  // For each loop `person` will contain the current `{ ... }` item.

  // Because we want add this property to the existant object, we don't need a new object. And as you said,
  // 1. `ages` will be deleted in each loop because is defined into `forEach`
  // var ages = {};

      // We get the century of die.
  var century = Math.ceil(person.died / 100),
      // and age of die
      age = person.died - person.born;

  // now if the century property not exist, create a new array into to accept all centuries after.
  if (!ages[century]) {
      ages[century] = [];
  }
  // equivalent to `!ages[century] && (ages[century] = []);`

    // in all cases, just push the value to the end.
    ages[century].push(age);
});

// Still exist !
console.log(ages); // Object {18: Array(2), 19: Array(1), 20: Array(1)}

your final format is what you want:

{
    18: [39, 93],
    19: [83],
    20: [73]
}

you need to create array in else statement like this ages[century] = [person.died - person.born];

    ancestry.forEach(function(person) {
      var ages = {};
      var century = Math.ceil(person.died / 100);

      if (century in ages) 
        {
        ages[century].push(person.died - person.born);
            }
      else
           {       
        ages[century] = [person.died - person.born];
           }
    });

You need to define ages array outside.

The error you are getting because you have not initialised your ages[century] as an array. Hence it will just store one value.

add this line

ages[century] = []

ages[century].push() will work fine then

Here is the change you need to do in your if-else

if (ages[century]){
   ages[century].push(person.died - person.born);
}
else{
   ages[century] = []
   ages[century].push(person.died - person.born);
}

You need to instantiate the century variables as keys in a key-value pair (a type of array called an "associative array") and then store ages into another array within that array (remember century is your key and the value is an array).

Basically, it's a 2-dimensional associative array so you might as well just use an object.

So it needs to look like var ages = {16: {24, 51, 16}, 17: {73, 22}, 18: {54, 65, 28}};

or

var ages = {16: {}, 17: {}, 18: {}}; and then you can recursively push those values onto the appropriate key's array.

Key-Value Pairs: Best way to store a key=>value array in JavaScript?

Multidimensional Associative Arrays (objects): Multi-dimensional associative arrays in javascript

W3School (JavaScript Objects): https://www.w3schools./js/js_objects.asp

You could just store the array of ages in an object and access that array by the key (which in your case you have declared to be century).

If it were me, I would make person an object and then store each person's birthday, date-of-death, and age-at-death as an attribute in a database.

Out of curiosity, what do you do when someone lives in two different centuries? I was born in 1987 and it's 2017 so did I live in the 20th or the 21st century?

发布者:admin,转转请注明出处:http://www.yc00.com/questions/1743701535a4492658.html

相关推荐

  • javascript - Pushing to an array inside an object - Stack Overflow

    I just recently picked up JS and have been working on an exercise to sort people from an array of recor

    17小时前
    10

发表回复

评论列表(0条)

  • 暂无评论

联系我们

400-800-8888

在线咨询: QQ交谈

邮件:admin@example.com

工作时间:周一至周五,9:30-18:30,节假日休息

关注微信