Сортировать массив по дате ISO 8601

как я могу отсортировать этот массив по дате (ISO 8601)?

var myArray = new Array();

myArray[0] = { name:'oldest', date:'2007-01-17T08:00:00Z' }
myArray[1] = { name:'newest', date:'2011-01-28T08:00:00Z' }
myArray[2] = { name:'old',    date:'2009-11-25T08:00:00Z' }

Детская площадка:
http://jsfiddle.net/4tUZt/

Заранее спасибо!

Ответ 1

Сортировать лексикографически:

Как отмечает @kdbanman, ISO8601 See Общие принципы были разработаны для лексикографического вида. Таким образом, строковое представление ISO8601 может быть отсортировано как любая другая строка, и это даст ожидаемый порядок.

'2007-01-17T08:00:00Z' < '2008-01-17T08:00:00Z' === true

Итак, вы бы реализовали:

var myArray = [
    { name:'oldest', date:'2007-01-17T08:00:00Z' },
    { name:'newest', date:'2011-01-28T08:00:00Z' },
    { name:'old',    date:'2009-11-25T08:00:00Z' }
];

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

Сортировка с использованием JavaScript Дата:

Более старые версии WebKit и Internet Explorer не поддерживают даты ISO 8601, поэтому вы должны сделать совместимую дату. Он поддерживается FireFox и современным WebKit. Более подробную информацию о поддержке Date.parse смотрите здесь. JavaScript: Какие браузеры поддерживают разбор строки даты ISO-8601 с Date.parse

Вот очень хорошая статья для создания даты, совместимой с Javascript ISO 8601, которую вы можете затем отсортировать как обычные даты javascript.

http://webcloud.se/log/JavaScript-and-ISO-8601/

Date.prototype.setISO8601 = function (string) {
    var regexp = "([0-9]{4})(-([0-9]{2})(-([0-9]{2})" +
    "(T([0-9]{2}):([0-9]{2})(:([0-9]{2})(\.([0-9]+))?)?" +
    "(Z|(([-+])([0-9]{2}):([0-9]{2})))?)?)?)?";
    var d = string.match(new RegExp(regexp));

    var offset = 0;
    var date = new Date(d[1], 0, 1);

    if (d[3]) { date.setMonth(d[3] - 1); }
    if (d[5]) { date.setDate(d[5]); }
    if (d[7]) { date.setHours(d[7]); }
    if (d[8]) { date.setMinutes(d[8]); }
    if (d[10]) { date.setSeconds(d[10]); }
    if (d[12]) { date.setMilliseconds(Number("0." + d[12]) * 1000); }
    if (d[14]) {
        offset = (Number(d[16]) * 60) + Number(d[17]);
        offset *= ((d[15] == '-') ? 1 : -1);
    }

    offset -= date.getTimezoneOffset();
    time = (Number(date) + (offset * 60 * 1000));
    this.setTime(Number(time));
}

Использование:

console.log(myArray.sort(sortByDate));  

function sortByDate( obj1, obj2 ) {
    var date1 = (new Date()).setISO8601(obj1.date);
    var date2 = (new Date()).setISO8601(obj2.date);
    return date2 > date1 ? 1 : -1;
}

Обновлено использование, чтобы включить метод сортировки credit @nbrooks

Ответ 2

Я бы пошел с этим:

console.clear();

var myArray = new Array();

myArray[0] = { name:'oldest', date:'2007-01-17T08:00:00Z' }
myArray[1] = { name:'newest', date:'2011-01-28T08:00:00Z' }
myArray[2] = { name:'old',    date:'2009-11-25T08:00:00Z' }

console.dir(myArray);

var newArray = myArray.sort(function(a,b){ 
    if (a.date < b.date) return -1; 
    else if (a.date > b.date) return 1; 
    else return 0;
});

console.dir(newArray);

Ответ 3

http://jsfiddle.net/4tUZt/2/

$(document).ready(function()
{ 
    var myArray = [ { name:'oldest', date:'2007-01-17T08:00:00Z' },
        { name:'newest', date:'2011-01-28T08:00:00Z' },
        { name:'old',    date:'2009-11-25T08:00:00Z' }];

    console.log( myArray.sort(sortByDate) );        
});

// Stable, ascending sort (use < for descending)
function sortByDate( obj1, obj2 ) {
    return new Date(obj2.date) > new Date(obj1.date) ? 1 : -1;
}

Ответ 4

Вы можете избежать создания дат и с помощью встроенной в лексикографическом сравнить функции String.prototype.localeCompare, а не ?: составной оператор или другие выражения:

var myArray = [
  {name: 'oldest', date: '2007-01-17T08:00:00Z'},
  {name: 'newest', date: '2011-01-28T08:00:00Z'},
  {name: 'old', date: '2009-11-25T08:00:00Z'}
];

// Oldest first
console.log(
  myArray.sort((a, b) => a.date.localeCompare(b.date))
);

// Newest first
console.log(
  myArray.sort((a, b) => -a.date.localeCompare(b.date))
);

Ответ 5

Демо: http://jsfiddle.net/4tUZt/4/

var myArray = new Array();

myArray[0] = { name:'oldest', date: '2007-01-17T08:00:00Z' };
myArray[1] = { name:'newest', date: '2011-01-28T08:00:00Z' };
myArray[2] = { name:'old',    date: '2009-11-25T08:00:00Z' };

var sortFunction = function (a, b) {
  return Date.parse(b.date) - Date.parse(a.date);
};

/* or

var sortFunction = function (a, b) {
  return new Date(b.date) - new Date(a.date);
};

*/

console.log(myArray.sort(sortFunction));

Ответ 6

ISO8601 предназначен для правильной сортировки в виде обычного текста, поэтому в общем случае будет выполняться обычная сортировка.

Чтобы сортировать по определенному ключу объектов в массиве, вам нужно указать функцию сравнения для метода sort(). На многих других языках их легко написать с помощью функции cmp, но JS не имеет встроенной функции cmp, поэтому мне легче всего написать мою собственную.

var myArray = new Array();

myArray[0] = { name:'oldest', date:'2007-01-17T08:00:00Z' }
myArray[1] = { name:'newest', date:'2011-01-28T08:00:00Z' }
myArray[2] = { name:'old',    date:'2009-11-25T08:00:00Z' }

// cmp helper function - built in to many other languages
var cmp = function (a, b) {
    return (a > b) ? 1 : ( (a > b) ? -1 : 0 );
}

myArray.sort(function (a,b) { return cmp(a.date, b.date) });

Ps Я бы написал свой массив, используя синтаксис JSON, например:

var myArray = [
    { name:'oldest', date:'2007-01-17T08:00:00Z' },
    { name:'newest', date:'2011-01-28T08:00:00Z' },
    { name:'old',    date:'2009-11-25T08:00:00Z' }
];

Ответ 7

Будьте осторожны, принятый ответ теперь советует отсортировать наши даты лексикографически.

Тем не менее, это будет работать, только если все ваши строки используют часовой пояс 'Z' или '+00' (= UTC). Строки даты, заканчивающиеся на "Z", соответствуют стандарту ISO8601, но все ISO8601 не заканчиваются на "Z".

Таким образом, чтобы быть полностью совместимым с ISO8601, вам необходимо проанализировать ваши строки с некоторой библиотекой Date (например, Javascript Date или Moment.js) и сравнить эти объекты. В этой части вы можете проверить ответ Скотта, который также охватывает браузеры, несовместимые с ISO8601.

Мой простой пример с Javascript Date (работает в любом не слишком старом браузере):

var myArray = [
    { name:'oldest', date:'2007-01-17T08:00:00Z' },
    { name:'newest', date:'2011-01-28T08:00:00+0100' },
    { name:'old',    date:'2009-11-25T08:00:00-0100' }
];

myArray.sort(function(a, b) {
    return new Date(a.date) - new Date(b.date);
});

Недостаток: это медленнее, чем просто сравнивать строки лексикографически.

Больше информации о стандарте ISO8601: здесь.