Я думаю, що багато людей хочуть зробити розбір рядків дати JSON. Існує хороший шанс, якщо ви перейдете на цю сторінку, що ви, можливо, захочете перетворити дату JavaScript JSON в дату Java.
Щоб показати, як виглядає рядок дати JSON:
var d=new Date();
var s = JSON.stringify(d);
document.write(s);
document.write("<br />"+d);
"2013-12-14T01:55:33.412Z"
Fri Dec 13 2013 17:55:33 GMT-0800 (PST)
Рядок дати JSON - 2013-12-14T01: 55: 33.412Z.
Дані не поширюються на специфікацію JSON на приклад, але вищезазначене - це дуже специфічний формат ISO 8601, тоді як ISO_8601 набагато більший, і це просто підмножина, хоча й дуже важлива.
Дивіться http://www.json.org
Дивіться http://en.wikipedia.org/wiki/ISO_8601
Див. Http://www.w3.org/TR/NOTE-datetime
Як це буває, я написав аналізатор JSON і аналізатор PLIST, обидва з яких використовують ISO-8601, але не однакові біти.
/*
var d=new Date();
var s = JSON.stringify(d);
document.write(s);
document.write("<br />"+d);
"2013-12-14T01:55:33.412Z"
Fri Dec 13 2013 17:55:33 GMT-0800 (PST)
*/
@Test
public void jsonJavaScriptDate() {
String test = "2013-12-14T01:55:33.412Z";
Date date = Dates.fromJsonDate ( test );
Date date2 = Dates.fromJsonDate_ ( test );
assertEquals(date2.toString (), "" + date);
puts (date);
}
Я написав два способи зробити це для свого проекту. Один стандарт, один швидкий.
Знову ж таки, рядок дати JSON - це дуже специфічна реалізація ISO 8601 ....
(Я розмістив іншу відповідь в іншій відповіді, яка повинна працювати для дат PLIST, які мають інший формат ISO 8601).
Дата JSON така:
public static Date fromJsonDate_( String string ) {
try {
return new SimpleDateFormat ( "yyyy-MM-dd'T'HH:mm:ss.SSSXXX").parse ( string );
} catch ( ParseException e ) {
return Exceptions.handle (Date.class, "Not a valid JSON date", e);
}
}
Для файлів PLIST (ASCII non GNUNext) також використовується ISO 8601, але немає мілісекунд, тому ... не всі дати ISO-8601 однакові. (Принаймні, я ще не знайшов такого, який використовує міліс, і аналізатор, який я бачив, зовсім пропускає часовий пояс OMG).
Тепер про швидку версію (ви можете знайти її в Boon).
public static Date fromJsonDate( String string ) {
return fromJsonDate ( Reflection.toCharArray ( string ), 0, string.length () );
}
Зауважте, що Reflection.toCharArray використовує небезпечні, якщо вони доступні, але за замовчуванням до string.toCharArray якщо ні.
(Ви можете взяти його з прикладу, замінивши Reflection.toCharArray (string) на string.toCharArray ()).
public static Date fromJsonDate( char[] charArray, int from, int to ) {
if (isJsonDate ( charArray, from, to )) {
int year = CharScanner.parseIntFromTo ( charArray, from + 0, from + 4 );
int month = CharScanner.parseIntFromTo ( charArray, from +5, from +7 );
int day = CharScanner.parseIntFromTo ( charArray, from +8, from +10 );
int hour = CharScanner.parseIntFromTo ( charArray, from +11, from +13 );
int minute = CharScanner.parseIntFromTo ( charArray, from +14, from +16 );
int second = CharScanner.parseIntFromTo ( charArray, from +17, from +19 );
int miliseconds = CharScanner.parseIntFromTo ( charArray, from +20, from +23 );
TimeZone tz = TimeZone.getTimeZone ( "GMT" );
return toDate ( tz, year, month, day, hour, minute, second, miliseconds );
} else {
return null;
}
}
IsJsonDate реалізується наступним чином:
public static boolean isJsonDate( char[] charArray, int start, int to ) {
boolean valid = true;
final int length = to -start;
if (length != JSON_TIME_LENGTH) {
return false;
}
valid &= (charArray [ start + 19 ] == '.');
if (!valid) {
return false;
}
valid &= (charArray[ start +4 ] == '-') &&
(charArray[ start +7 ] == '-') &&
(charArray[ start +10 ] == 'T') &&
(charArray[ start +13 ] == ':') &&
(charArray[ start +16 ] == ':');
return valid;
}
У будь-якому випадку ... я здогадуюсь, що досить багато людей, які приїжджають сюди .., можливо, шукають рядок дати JSON, і хоча це дата ISO-8601, це дуже специфічний, який потребує дуже конкретного розбору.
public static int parseIntFromTo ( char[] digitChars, int offset, int to ) {
int num = digitChars[ offset ] - '0';
if ( ++offset < to ) {
num = ( num * 10 ) + ( digitChars[ offset ] - '0' );
if ( ++offset < to ) {
num = ( num * 10 ) + ( digitChars[ offset ] - '0' );
if ( ++offset < to ) {
num = ( num * 10 ) + ( digitChars[ offset ] - '0' );
if ( ++offset < to ) {
num = ( num * 10 ) + ( digitChars[ offset ] - '0' );
if ( ++offset < to ) {
num = ( num * 10 ) + ( digitChars[ offset ] - '0' );
if ( ++offset < to ) {
num = ( num * 10 ) + ( digitChars[ offset ] - '0' );
if ( ++offset < to ) {
num = ( num * 10 ) + ( digitChars[ offset ] - '0' );
if ( ++offset < to ) {
num = ( num * 10 ) + ( digitChars[ offset ] - '0' );
}
}
}
}
}
}
}
}
return num;
}
Дивіться https://github.com/RichardHightower/boon
Boon має аналізатор PLIST (ASCII) та аналізатор JSON.
Парсер JSON - це найшвидший синтаксичний аналізатор Java JSON, про який я знаю.
Незалежно перевірені хлопцями з виступу Gatling Performance.
https://github.com/gatling/json-parsers-benchmark
Benchmark Mode Thr Count Sec Mean Mean error Units
BoonCharArrayBenchmark.roundRobin thrpt 16 10 1 724815,875 54339,825 ops/s
JacksonObjectBenchmark.roundRobin thrpt 16 10 1 580014,875 145097,700 ops/s
JsonSmartBytesBenchmark.roundRobin thrpt 16 10 1 575548,435 64202,618 ops/s
JsonSmartStringBenchmark.roundRobin thrpt 16 10 1 541212,220 45144,815 ops/s
GSONStringBenchmark.roundRobin thrpt 16 10 1 522947,175 65572,427 ops/s
BoonDirectBytesBenchmark.roundRobin thrpt 16 10 1 521528,912 41366,197 ops/s
JacksonASTBenchmark.roundRobin thrpt 16 10 1 512564,205 300704,545 ops/s
GSONReaderBenchmark.roundRobin thrpt 16 10 1 446322,220 41327,496 ops/s
JsonSmartStreamBenchmark.roundRobin thrpt 16 10 1 276399,298 130055,340 ops/s
JsonSmartReaderBenchmark.roundRobin thrpt 16 10 1 86789,825 17690,031 ops/s
Він має найшвидший аналізатор JSON для потоків, читачів, байтів [], char [], CharSequence (StringBuilder, CharacterBuffer) та String.
Дивіться більше орієнтирів на:
https://github.com/RichardHightower/json-parsers-benchmark