【问题标题】:How to sort an object array by date property?如何按日期属性对对象数组进行排序?
【发布时间】:2012-04-24 19:42:48
【问题描述】:

假设我有一个包含几个对象的数组:

var array = [{id: 1, date: Mar 12 2012 10:00:00 AM}, {id: 2, date: Mar 8 2012 08:00:00 AM}];

如何按日期元素对这个数组进行排序,从最接近当前日期和时间的日期开始排序?请记住,数组可能有很多对象,但为了简单起见,我使用了 2。

我会使用排序功能和自定义比较器吗?

【问题讨论】:

  • 如果你使用 Date 构造函数,请先检查这个stackoverflow.com/questions/5619202/…
  • 最快的方法是使用同构的sort-array 模块,它可以在浏览器和节点中本地工作,支持任何类型的输入、计算字段和自定义排序顺序。
  • 这个问题好像没有人回答。以下答案均未解释如何“从最接近当前日期的日期开始订购”。

标签: javascript datetime


【解决方案1】:

最简单的答案

array.sort(function(a,b){
  // Turn your strings into dates, and then subtract them
  // to get a value that is either negative, positive, or zero.
  return new Date(b.date) - new Date(a.date);
});

更通用的答案

array.sort(function(o1,o2){
  if (sort_o1_before_o2)    return -1;
  else if(sort_o1_after_o2) return  1;
  else                      return  0;
});

或者更简洁:

array.sort(function(o1,o2){
  return sort_o1_before_o2 ? -1 : sort_o1_after_o2 ? 1 : 0;
});

通用、强大的答案

在所有数组上使用Schwartzian transform 定义一个自定义的不可枚举的sortBy 函数:

(function(){
  if (typeof Object.defineProperty === 'function'){
    try{Object.defineProperty(Array.prototype,'sortBy',{value:sb}); }catch(e){}
  }
  if (!Array.prototype.sortBy) Array.prototype.sortBy = sb;

  function sb(f){
    for (var i=this.length;i;){
      var o = this[--i];
      this[i] = [].concat(f.call(o,o,i),o);
    }
    this.sort(function(a,b){
      for (var i=0,len=a.length;i<len;++i){
        if (a[i]!=b[i]) return a[i]<b[i]?-1:1;
      }
      return 0;
    });
    for (var i=this.length;i;){
      this[--i]=this[i][this[i].length-1];
    }
    return this;
  }
})();

像这样使用它:

array.sortBy(function(o){ return o.date });

如果您的日期不能直接比较,请从中制作一个可比较的日期,例如

array.sortBy(function(o){ return new Date( o.date ) });

如果您返回一个值数组,您也可以使用它来按多个条件排序:

// Sort by date, then score (reversed), then name
array.sortBy(function(o){ return [ o.date, -o.score, o.name ] };

详情请见http://phrogz.net/JS/Array.prototype.sortBy.js

【讨论】:

  • 为什么不在简单答案中使用return b-a;
  • 不推荐在 sort 方法中创建新的 Date 对象。为此专门遇到了生产性能问题。不要在排序方法中分配内存(和 GC)。
  • 第一个 eg 语法在 angular7 上给出错误:算术运算的左侧必须是 'any'、'number'、'bigint' 类型或枚举类型
  • @Sireini 取决于日期属性的类型/格式,但如果它是标准的“日期”对象,那么@Gal 的响应是最快的并且没有分配。 a-b 评论在 Chrome (jsperf.com/date-sort-mm/1) 中实际上非常慢。如果日期只是 ISO 字符串,最好只比较它们a &gt; b ? 1 : a &lt; b ? -1 : 0
  • 如果你正在使用 TS 并且想按日期属性对对象进行排序,你可以使用第一个选项,如下所示:return +b.date - +a.date; 用于降序,并颠倒 'a' 和 'b 的顺序' 升序
【解决方案2】:

@Phrogz 的答案都很棒,但这里有一个很棒、更简洁的答案:

array.sort(function(a,b){return a.getTime() - b.getTime()});

使用箭头函数方式

array.sort((a,b)=>a.getTime()-b.getTime());

在这里找到:Sort date in Javascript

【讨论】:

  • 直接计算 a - b 也可以。所以,array.sort((a, b) =&gt; a - b) (es6)
  • a.getTime() - b.getTime()a - b 快得多
【解决方案3】:

在更正 JSON 之后,现在应该可以为您工作了:

var array = [{id: 1, date:'Mar 12 2012 10:00:00 AM'}, {id: 2, date:'Mar 8 2012 08:00:00 AM'}];


array.sort(function(a, b) {
    var c = new Date(a.date);
    var d = new Date(b.date);
    return c-d;
});

【讨论】:

    【解决方案4】:

    您的数据需要更正:

    var array = [{id: 1, date: "Mar 12 2012 10:00:00 AM"},{id: 2, date: "Mar 28 2012 08:00:00 AM"}];
    

    修正数据后,就可以使用这段代码了:

    function sortFunction(a,b){  
        var dateA = new Date(a.date).getTime();
        var dateB = new Date(b.date).getTime();
        return dateA > dateB ? 1 : -1;  
    }; 
    
    var array = [{id: 1, date: "Mar 12 2012 10:00:00 AM"},{id: 2, date: "Mar 28 2012 08:00:00 AM"}];
    array.sort(sortFunction);​
    

    【讨论】:

    • 对于使用 Typescript 的任何人,我都可以使用此功能按日期排序,而其他使用日期减法的人则失败。
    • 有什么区别?我看到了不同类型的引号和空格,但这没关系,还是我遗漏了什么?
    【解决方案5】:

    我推荐GitHub: Array sortBy - 使用Schwartzian transformsortBy 方法的最佳实现

    但现在我们将尝试这种方法Gist: sortBy-old.js
    让我们创建一个方法来对能够按某些属性排列对象的数组进行排序。

    创建排序函数

    var sortBy = (function () {
      var toString = Object.prototype.toString,
          // default parser function
          parse = function (x) { return x; },
          // gets the item to be sorted
          getItem = function (x) {
            var isObject = x != null && typeof x === "object";
            var isProp = isObject && this.prop in x;
            return this.parser(isProp ? x[this.prop] : x);
          };
          
      /**
       * Sorts an array of elements.
       *
       * @param {Array} array: the collection to sort
       * @param {Object} cfg: the configuration options
       * @property {String}   cfg.prop: property name (if it is an Array of objects)
       * @property {Boolean}  cfg.desc: determines whether the sort is descending
       * @property {Function} cfg.parser: function to parse the items to expected type
       * @return {Array}
       */
      return function sortby (array, cfg) {
        if (!(array instanceof Array && array.length)) return [];
        if (toString.call(cfg) !== "[object Object]") cfg = {};
        if (typeof cfg.parser !== "function") cfg.parser = parse;
        cfg.desc = !!cfg.desc ? -1 : 1;
        return array.sort(function (a, b) {
          a = getItem.call(cfg, a);
          b = getItem.call(cfg, b);
          return cfg.desc * (a < b ? -1 : +(a > b));
        });
      };
      
    }());
    

    设置未排序的数据

    var data = [
      {date: "2011-11-14T17:25:45Z", quantity: 2, total: 200, tip: 0,   type: "cash"},
      {date: "2011-11-14T16:28:54Z", quantity: 1, total: 300, tip: 200, type: "visa"},
      {date: "2011-11-14T16:30:43Z", quantity: 2, total: 90,  tip: 0,   type: "tab"},
      {date: "2011-11-14T17:22:59Z", quantity: 2, total: 90,  tip: 0,   type: "tab"},
      {date: "2011-11-14T16:53:41Z", quantity: 2, total: 90,  tip: 0,   type: "tab"},
      {date: "2011-11-14T16:48:46Z", quantity: 2, total: 90,  tip: 0,   type: "tab"},
      {date: "2011-11-31T17:29:52Z", quantity: 1, total: 200, tip: 100, type: "visa"},
      {date: "2011-11-01T16:17:54Z", quantity: 2, total: 190, tip: 100, type: "tab"},
      {date: "2011-11-14T16:58:03Z", quantity: 2, total: 90,  tip: 0,   type: "tab"},
      {date: "2011-11-14T16:20:19Z", quantity: 2, total: 190, tip: 100, type: "tab"},
      {date: "2011-11-14T17:07:21Z", quantity: 2, total: 90,  tip: 0,   type: "tab"},
      {date: "2011-11-14T16:54:06Z", quantity: 1, total: 100, tip: 0,   type: "cash"}
    ];
    

    使用它

    最后,我们通过"date"属性将数组排列为string

    //sort the object by a property (ascending)
    //sorting takes into account uppercase and lowercase
    sortBy(data, { prop: "date" });
    

    如果你想忽略字母大小写,设置"parser"回调:

    //sort the object by a property (descending)
    //sorting ignores uppercase and lowercase
    sortBy(data, {
        prop: "date",
        desc: true,
        parser: function (item) {
            //ignore case sensitive
            return item.toUpperCase();
        }
    });
    

    如果您想将“日期”字段视为Date,请键入:

    //sort the object by a property (ascending)
    //sorting parses each item to Date type
    sortBy(data, {
        prop: "date",
        parser: function (item) {
            return new Date(item);
        }
    });
    

    这里你可以玩一下上面的例子:
    jsbin.com/lesebi

    【讨论】:

    • IE11 的行有问题: if (toString.call(cfg) !== "[object Object]") cfg = {};如果将其替换为 if (Object.prototype.toString.call(cfg) !== "[object Object]") cfg = {};你也会对 IE11 很好。
    【解决方案6】:

    当您的日期采用这种格式 (dd/mm/yyyy) 时应该这样做。

      sortByDate(arr) {
        arr.sort(function(a,b){
          return Number(new Date(a.readableDate)) - Number(new Date(b.readableDate));
        });
    
        return arr;
      }
    

    然后拨打sortByDate(myArr);

    【讨论】:

      【解决方案7】:

      您可以在下划线 js 中使用 sortBy。

      http://underscorejs.org/#sortBy

      示例:

      var log = [{date: '2016-01-16T05:23:38+00:00', other: 'sample'}, 
                 {date: '2016-01-13T05:23:38+00:00',other: 'sample'}, 
                 {date: '2016-01-15T11:23:38+00:00', other: 'sample'}];
      
      console.log(_.sortBy(log, 'date'));
      

      【讨论】:

        【解决方案8】:

        arr 是一个对象数组,每个对象都有 date_prop ,它是一个日期。您可以像这样按降序/降序对其进行排序

         arr = arr.sort(function (a, b) {
              var dateA = new Date(a.date_prop).getTime();
              var dateB = new Date(b.date_prop).getTime();
              return dateA < dateB ? 1 : -1; // ? -1 : 1 for ascending/increasing order
            });
        

        【讨论】:

          【解决方案9】:

          以上答案都很好?,这是我以 ES6 方式排序日期的实现,我使用Date.parse(是全局日期对象)这会将日期的字符串表示形式转换为毫秒数。而不是每次都实例化new Date 对象。

          var array = ["2021-08-10T07:24:30.087+0000" , "2021-09-30T07:24:30.087+0000", "2021-10-13T07:24:30.087+0000"];
          
          // sorting with latest date
          array.sort((a,b) => Date.parse(b) - Date.parse(a))
          

          【讨论】:

          • 答案 sn-p 不适合这个问题。您可以更新您的答案以匹配问题或将其删除。
          • 谢谢。它正确排序所有图像
          【解决方案10】:

          我将在此处添加此内容,因为某些用途可能无法解决如何反转此排序方法。

          要按“上来”排序,我们可以简单地交换 a 和 b,如下所示:

          your_array.sort ( (a, b) => {
                return new Date(a.DateTime) - new Date(b.DateTime);
          });
          

          注意a 现在在左侧,b 在右侧,:D!

          【讨论】:

            【解决方案11】:

            使用 ES6 箭头函数,您可以进一步编写一行简洁的代码(不包括变量声明)。

            例如:

            var isDescending = true; //set to false for ascending
            console.log(["8/2/2020","8/1/2020","8/13/2020", "8/2/2020"].sort((a,b) => isDescending ? new Date(b).getTime() - new Date(a).getTime() : new Date(a).getTime() - new Date(b).getTime()));

            由于上述日期不存在时间,Date 对象将考虑以下默认时间进行排序:

            00:00:00

            代码适用于升序和降序排序。 只需根据需要更改isDescending 变量的值即可。

            【讨论】:

              【解决方案12】:

              我个人使用以下方法对日期进行排序。

              let array = ["July 11, 1960", "February 1, 1974", "July 11, 1615", "October 18, 1851", "November 12, 1995"];
              
              array.sort(function(date1, date2) {
                 date1 = new Date(date1);
                 date2 = new Date(date2);
                 if (date1 > date2) return 1;
                 if (date1 < date2) return -1;
              })
              

              【讨论】:

                【解决方案13】:

                我能够使用以下行实现排序:

                array.sort(function(a, b)
                {
                   if (a.DueDate > b.DueDate) return 1;
                   if (a.DueDate < b.DueDate) return -1;
                })
                

                【讨论】:

                  【解决方案14】:
                  Adding absolute will give better results
                  
                  var datesArray =[
                        {"some":"data1","date": "2018-06-30T13:40:31.493Z"},
                        {"some":"data2","date": "2018-07-04T13:40:31.493Z"},
                        {"some":"data3","date": "2018-06-27T13:40:54.394Z"}
                     ]
                  
                  var sortedJsObjects = datesArray.sort(function(a,b){ 
                      return Math.abs(new Date(a.date) - new Date(b.date)) 
                  });
                  

                  【讨论】:

                  • 与当前日期的比较在哪里?
                  • 这根本行不通。
                  【解决方案15】:

                  带有日期的字符串在 JavaScript 中是可比较的(如果它们在语法上相同),例如:

                  '2020-12-01' < '2020-12-02' == true
                  

                  这意味着您可以在自定义排序函数中使用此表达式:

                  var arr = [{id:1, date:'2020-12-01'}, {id:1, date:'2020-12-15'}, {id:1, date:'2020-12-12'}]
                  
                  function sortByDate(a, b) {
                      if (a.date < b.date) {
                          return 1;
                      }
                      if (a.date > b.date) {
                          return -1;
                      }
                      return 0;
                  }
                  
                  const sorted = arr.sort(sortByDate);
                  console.log(sorted);

                  【讨论】:

                    【解决方案16】:

                    谢谢 Ganesh Sanap。按日期字段从旧到新对项目进行排序。使用它

                     myArray = [{transport: "Air",
                                 load: "Vatican Vaticano",
                                 created: "01/31/2020"},
                                {transport: "Air",
                                 load: "Paris",
                                 created: "01/30/2020"}] 
                    
                            myAarray.sort(function(a, b) {
                                var c = new Date(a.created);
                                var d = new Date(b.created);
                                return c-d;
                            });
                    

                    【讨论】:

                    【解决方案17】:

                    对于任何想要按日期(英国格式)排序的人,我使用了以下内容:

                    //Sort by day, then month, then year
                    for(i=0;i<=2; i++){
                        dataCourses.sort(function(a, b){
                    
                            a = a.lastAccessed.split("/");
                            b = b.lastAccessed.split("/");
                    
                            return a[i]>b[i] ? -1 : a[i]<b[i] ? 1 : 0;
                        }); 
                    }
                    

                    【讨论】:

                      【解决方案18】:

                      我刚刚采用了上面描述的Schwartzian transform 并将其作为函数。它需要一个array、排序function 和一个布尔值作为输入:

                      function schwartzianSort(array,f,asc){
                          for (var i=array.length;i;){
                            var o = array[--i];
                            array[i] = [].concat(f.call(o,o,i),o);
                          }
                          array.sort(function(a,b){
                            for (var i=0,len=a.length;i<len;++i){
                              if (a[i]!=b[i]) return a[i]<b[i]?asc?-1:1:1;
                            }
                            return 0;
                          });
                          for (var i=array.length;i;){
                            array[--i]=array[i][array[i].length-1];
                          }
                          return array;
                        }
                      

                      function schwartzianSort(array, f, asc) {
                        for (var i = array.length; i;) {
                          var o = array[--i];
                          array[i] = [].concat(f.call(o, o, i), o);
                        }
                        array.sort(function(a, b) {
                          for (var i = 0, len = a.length; i < len; ++i) {
                            if (a[i] != b[i]) return a[i] < b[i] ? asc ? -1 : 1 : 1;
                          }
                          return 0;
                        });
                        for (var i = array.length; i;) {
                          array[--i] = array[i][array[i].length - 1];
                        }
                        return array;
                      }
                      
                      arr = []
                      arr.push({
                        date: new Date(1494434112806)
                      })
                      arr.push({
                        date: new Date(1494434118181)
                      })
                      arr.push({
                        date: new Date(1494434127341)
                      })
                      
                      console.log(JSON.stringify(arr));
                      
                      arr = schwartzianSort(arr, function(o) {
                        return o.date
                      }, false)
                      console.log("DESC", JSON.stringify(arr));
                      
                      arr = schwartzianSort(arr, function(o) {
                        return o.date
                      }, true)
                      console.log("ASC", JSON.stringify(arr));

                      【讨论】:

                        【解决方案19】:
                        ["12 Jan 2018" , "1 Dec 2018", "04 May 2018"].sort(function(a,b) {
                            return new Date(a).getTime() - new Date(b).getTime()
                        })
                        

                        【讨论】:

                        • 请简要说明您的答案并检查您的代码格式。
                        • 旧日期将失败。
                        【解决方案20】:

                        如果您像我一样有一个日期格式为 YYYY[-MM[-DD]] 的数组,您想在不太具体的日期之前订购更具体的日期,我想出了这个方便的功能:

                        function sortByDateSpecificity(a, b) {
                          const aLength = a.date.length
                          const bLength = b.date.length
                          const aDate = a.date + (aLength < 10 ? '-12-31'.slice(-10 + aLength) : '')
                          const bDate = b.date + (bLength < 10 ? '-12-31'.slice(-10 + bLength) : '')
                          return new Date(aDate) - new Date(bDate)
                        }
                        

                        【讨论】:

                          【解决方案21】:

                          感谢上面那些精彩的答案。我想到了一个稍微复杂的答案。仅适用于那些想要比较不同答案的人。

                          const data = [
                              '2-2018', '1-2018',
                              '3-2018', '4-2018',
                              '1-2019', '2-2019',
                              '3-2019', '4-2019',
                              '1-2020', '3-2020',
                              '4-2020', '1-2021'
                          ]
                          
                          let eachYearUniqueMonth = data.reduce((acc, elem) => {
                              const uniqueDate = Number(elem.match(/(\d+)\-(\d+)/)[1])
                              const uniqueYear = Number(elem.match(/(\d+)\-(\d+)/)[2])
                          
                          
                              if (acc[uniqueYear] === undefined) {
                                  acc[uniqueYear] = []        
                              } else{    
                                 if (acc[uniqueYear]  && !acc[uniqueYear].includes(uniqueDate)) {
                                    acc[uniqueYear].push(uniqueDate)
                                }
                              }
                          
                              return acc;
                          }, {})
                          
                          
                          let group = Object.keys(eachYearUniqueMonth).reduce((acc,uniqueYear)=>{
                              eachYearUniqueMonth[uniqueYear].forEach(uniqueMonth=>{
                              acc.push(`${uniqueYear}-${uniqueMonth}`)
                            })
                            
                            return acc;
                          },[])
                          
                          console.log(group);   //["2018-1", "2018-3", "2018-4", "2019-2", "2019-3", "2019-4", "2020-3", "2020-4"]
                          
                          
                          

                          【讨论】:

                            【解决方案22】:

                            我有一个对象数组,其中键“时间”包含 ISO 格式的日期

                            示例 - 2021-12-24T11:02:20.370705

                            arr.sort(function(a,b){return a.time > b.time ? 1 : a.time < b.time ? -1 : 0 });
                            

                            以上对我来说就像一个魅力!

                            【讨论】:

                              猜你喜欢
                              • 1970-01-01
                              • 2023-03-22
                              • 2021-07-15
                              • 1970-01-01
                              • 2013-10-26
                              • 1970-01-01
                              • 2021-03-29
                              相关资源
                              最近更新 更多