如何按日期属性对对象数组进行排序?

IT技术 javascript datetime
2021-02-01 19:39:07

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

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

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

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

6个回答

最简单的答案

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;
});

通用、强大的答案

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

(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

第一个 eg 语法在 angular7 上给出错误:算术运算的左侧必须是“any”、“number”、“bigint”或枚举类型的类型
2021-03-16 19:39:07
如果您正在使用 TS 并希望按日期属性对对象进行排序,则可以使用第一个选项,如下所示:return +b.date - +a.date;用于降序,并反转 'a' 和 'b' 的顺序用于升序
2021-03-21 19:39:07
不建议在 sort 方法中创建新的 Date 对象。由于这个原因,已经遇到了生产性能问题。不要在排序方法中分配内存(和 GC)。
2021-03-23 19:39:07
为什么不在return b-a;简单答案中?
2021-04-06 19:39:07
@Sireini 取决于日期属性的类型/格式,但如果它是标准的“日期”对象,那么@Gal 的响应速度最快且无需分配。a-bChrome ( jsperf.com/date-sort-mm/1 ) 中评论实际上非常慢如果日期只是 ISO 字符串,那么只比较它们是最快的a > b ? 1 : a < b ? -1 : 0
2021-04-06 19:39:07

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

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

使用箭头函数方式

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

在这里找到:在 Javascript 中对日期进行排序

直接做数学a - b也行。所以,array.sort((a, b) => a - b)(es6)
2021-03-13 19:39:07

更正 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;
});

您的数据需要一些更正:

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 的任何人,我都可以使用此函数按日期排序,而其他使用日期减法的则失败。
2021-03-21 19:39:07

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

但现在我们将尝试这种方法要点: 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。
2021-04-04 19:39:07