Як розібрати 20180810T143000Z до time_t

Що таке найкоротший/найелегантніший спосіб (тобто використання існуючих функцій lib) для аналізу рядка у вигляді 20180810T143000Z до time_t ? Зауважте, що літерал завжди представляє часову мітку UTC.

Я почав аналізувати рядок і присвоювати значення struct tm * tm , щоб зробити mktime (tm) наприкінці. Це відчуває себе надто складним, хоча.

0
Яку часову бібліотеку ви використовуєте? Я копію кожен необхідний символ з індексом масиву в буфер і перетворюю його на рік, місяць, день, години, хвилини. Після цього я б використав бібліотеку, щоб перетворити її в time_t за кількість секунд, починаючи з 1970 року. Ви можете показати, що ви маєте? sscanf може працювати, але деякі вважають, що це не елегантна функція.
додано Автор Standback, джерело

6 Відповіді

Розбір рядка дійсно є єдиним шляхом. Однак є багато способів зробити це.

Моїм кращим методом є спочатку перевірити правильність формату, звернувши увагу на те, що T і Z знаходяться в потрібному місці:

if (timeString[8] == 'T' && timeString[15] == 'Z') {
    ... parse in here
}

І синтаксичний аналіз просто робить числа з множенням:

int year = (timeString[0] - '0') * 1000 +
           (timeString[1] - '0') * 100 +
           (timeString[2] - '0') * 10 +
           (timeString[3] - '0');

Можна очистити макроси, якщо хочете:

#define NUM(off, mult) ((timeString[(off)] - '0') * (mult))

Потім:

int year =   NUM(0, 1000) + NUM(1, 100) + NUM(2, 10) + NUM(3, 1);
int month =  NUM(4, 10)   + NUM(5, 1);
int day =    NUM(6, 10)   + NUM(7, 1);
int hour =   NUM(9, 10)   + NUM(10, 1);
int minute = NUM(11, 10)  + NUM(12, 1);
int second = NUM(13, 10)  + NUM(14, 1);

А потім, так, помістіть їх у struct tm (або безпосередньо призначте їх результати розрахунків без використання проміжних змінних) і зателефонуйте mktime() .

2
додано
@ MarcelStör Звичайно - якщо це те, що ви хочете зробити.
додано Автор Majenko, джерело
Я не можу це сказати. Це залежить від решти вашої програми. Відповідь може бути "нічого", або вона може бути "повідомляти користувачеві", вона "запитувати нову мітку часу", або будь-яку кількість інших речей.
додано Автор Majenko, джерело
@Juraj Так, але ви не включили жодної відповіді у відповідь, лише блок коду;)
додано Автор Majenko, джерело
@Majenko вибачте, я був трохи неспецифічним. Я все ще знайомий з Arduino (як ви можете сказати). Що я мав на увазі, що в більшості інших середовищ/платформ, я знайомий з я б, можливо, кинути виняток. Отже, я вважаю, що повернення -1 і обробка цього поза функцією зробить це більш надійним.
додано Автор Krunal Hingu, джерело
"Спочатку перевірте, чи правильний формат" - дійсна точка, але що б ви зробили в гілці else ?
додано Автор Krunal Hingu, джерело

Ось ще один спосіб конвертувати рядок мітки часу в time_t. Тут є декілька відмінних відповідей, але ви можете порівнювати розміри двійкових файлів. Цей ескіз становить 3884 байта (версія версії 1.0.6.2, GCC 4.2.1).

#include 
TimeElements myTimeElements;
char timeString[] = "20180810T143000Z";

void setup(){

  Serial.begin(9600);

  myTimeElements.Year = CalendarYrToTm((timeString[0] - '0') * 1000 + (timeString[1] - '0') * 100 + (timeString[2] - '0') * 10 + (timeString[3] - '0'));
  myTimeElements.Month = (timeString[4] - '0') * 10 + (timeString[5] - '0');
  myTimeElements.Day = (timeString[6] - '0') * 10 + (timeString[7] - '0');
  myTimeElements.Hour = (timeString[9] - '0') * 10 + (timeString[10] - '0');
  myTimeElements.Minute = (timeString[11] - '0') * 10 + (timeString[12] - '0');
  myTimeElements.Second = (timeString[13] - '0') * 10 + (timeString[14] - '0');

 //Assemble time elements into time_t.
  time_t t = makeTime(myTimeElements);

 //Print out the contents of "t" one "piece" at a time using the "time_t" functions.
  Serial.println(year(t));
  Serial.println(month(t));
  Serial.println(day(t));
  Serial.println(hour(t));
  Serial.println(minute(t));
  Serial.println(second(t));

}

void loop(){}
1
додано

Це не є повним без sscanf.

Я взяв ескіз по @Juraj і оголосив окремі цілі числа, щоб переконатися, що кожен% d буде співпадати з цілим числом.

#include 

void setup() {

  Serial.begin(115200);

  char buff[] = "20180810T143000Z";

  TimeElements tm;

  int yr, mnth, d, h, m, s;
  sscanf( buff, "%4d%2d%2dT%2d%2d%2dZ", &yr, &mnth, &d, &h, &m, &s);

  tm.Year = yr - 1970;
  tm.Month = mnth;
  tm.Day = d;
  tm.Hour = h;
  tm.Minute = m;
  tm.Second = s;

  time_t t = makeTime(tm);

  sprintf(buff, "%02d-%02d-%02d %02d:%02d:%02d", year(t), month(t), day(t), hour(t), minute(t), second(t));

  Serial.println(buff);
}

void loop() {
}

Марсель Штор, зараз є чотири хороших рішення. На мій погляд, вони однаково хороші.

1
додано
#include 

void setup() {

  Serial.begin(115200);

  char buff[] = "20180810T143000Z";
  for (int i = 0; i < sizeof(buff); i++) {
    buff[i] = buff[i] - '0';
  }
  int yr = buff[0] * 1000 + buff[1] * 100 + buff[2] * 10 + buff[3];
  if (yr > 99)
    yr = yr - 1970;
  else
    yr += 30;
  TimeElements tm;
  tm.Year = yr;
  tm.Month = buff[4] * 10 + buff[5];
  tm.Day = buff[6] * 10 + buff[7];
 //8 T
  tm.Hour = buff[9] * 10 + buff[10];
  tm.Minute = buff[11] * 10 + buff[12];
  tm.Second = buff[13] * 10 + buff[14];

  time_t t = makeTime(tm);

  sprintf(buff, "%02d%02d%02d %02d%02d%02d", year(t), month(t), day(t), hour(t), minute(t), second(t));

  Serial.println(buff);

}

void loop() {

}

Ви можете встановити TimeLib в Library Manager. Працює на всіх платформах Arduino.

1
додано
У buff [i ++] * 10 + buff [i ++] порядок оцінки i ++ не визначено. Він може працювати як слід, з будь-якою конкретною комбінацією версії та налаштувань компілятора, і збій на наступному. gcc 5.4 попереджає мене, що “операція на i may може бути невизначеною”. Правильний спосіб зробити це tm.Month = buff [i ++] * 10; tm.Month + = buff [i ++]; .
додано Автор Sprogz, джерело
@ MarcelStör: Те, що я написав для tm.Month , дійсний для кожного екземпляра, де i ++ з'являється більше одного разу в одному й тому ж виразі.
додано Автор Sprogz, джерело
Re «компілятор не має підстав оцінювати [...]»: він також не має підстав оцінювати їх у порядку, який ви очікуєте! Будь ласка, дізнайтеся про точки послідовності в C ++, перш ніж робити такі неправильні припущення. Зверніть увагу на те, що "між попередньою і наступною точкою послідовності об'єкт повинен зберігати значення, що зберігається, не більше одного разу шляхом оцінки виразу."
додано Автор Sprogz, джерело
змінив i ++ на константи. (Рік 4-й. Yabba dabba doo)
додано Автор Juraj, джерело
у цьому випадку використання i ++ хороше. компілятор не має жодних підстав оцінювати деякий субекспресію з i ++ пізніше у виразі перед деяким надвисловленням з i ++ раніше у виразі. Я зміню. Я бачив попередження, але для цього я мав обмежений час.
додано Автор Juraj, джерело
@ MarcelStör, я прочитав коментар Едгара, але цей код не є якимсь фрагментом, а повним робочим прикладом. Я не люблю редагувати його без тестування. І тепер я не можу його перевірити.
додано Автор Juraj, джерело
@EdgarBonet Я знаю, має загальний сенс. Ось чому запропонував редагувати. Врешті-решт, це було нормально, щоб отримати відхилення, тому що насправді постраждали багато інших ліній.
додано Автор Krunal Hingu, джерело
@EdgarBonet дякую, я просто запропонував цю зміну: arduino.stackexchange.com/review/suggested-edits/38715
додано Автор Krunal Hingu, джерело
Так, я подивився на strptime , але прийшов до висновку, що він, ймовірно, не працюватиме без роздільників. Тому я навіть не намагався.
додано Автор Krunal Hingu, джерело

Альтернативою моїй першій відповіді буде використання стандартних функцій C з time.h і sscanf. C функція strptime не може аналізувати мітку часу без роздільників. Але sscanf може розібрати ваш вхід.

#include 

void setup() {

  Serial.begin(115200);

  const char* buff = "20180810T143000Z";

  tm tms;
  sscanf(buff, "%04d%02d%02dT%02d%02d%02d", &(tms.tm_year), &(tms.tm_mon), &(tms.tm_mday), &(tms.tm_hour), &(tms.tm_min), &(tms.tm_sec));
  tms.tm_year -= 1900;
  tms.tm_mon -= 1;
  tms.tm_isdst = 0;
  time_t t = mktime(&tms);

  Serial.println(ctime(&t));
}

void loop() {
}

У AVR "% 02hhd" необхідно використовувати, оскільки відповідні члени у struct tm є int8_t.

1
додано
% D - це ціле число, також для мікроконтролера avr. Відмінність полягає в тому, що TimeLib TimeElements не використовує цілі числа, а байти.
додано Автор Standback, джерело
Так, для C time.h tm_year та інші елементи - це цілі числа. Також для мікроконтролера avr% d збігається з цілим числом. % D очікує "int", а не 2 або 4 байтну змінну.
додано Автор Standback, джерело
Я спробував це в декількох різних способів з різними даними на стеку, використовуючи arduino 1.8.5 з дошкою arduino uno. "Int" - це два байти, а "short int" також два байти, а "short short int" не існує. Формат "% d" працює з "int", і "% hd", схоже, виконує те саме, що і "% d". Формат "% hhd" читає підписаний байт. Все, як я його пам'ятаю, ніяких дивних речей. Якщо ви можете довести, що ви маєте рацію з ескізом, чи можете ви розпочати нову тему для цього?
додано Автор Standback, джерело
Тепер я бачу, що tm для avr має елементи int8_t і int16_t. На жаль, вони дійсно не всі цілі числа для avr. Я не знаю, що. Дякую. Чи згодні ви, що% d в sscanf відповідає 'int' для avr? Я бачу, що ви зняли це заплутане останнє речення.
додано Автор Standback, джерело
% d на AVR працює для sprintf, але sscanf читати неправильно без hh. Спробуй це
додано Автор Juraj, джерело
@Jot, це C time.h
додано Автор Juraj, джерело
@Jot, це тільки тоді, коли я використовую члени структури tm як параметри sscanf. Попередження '% d' очікує аргумент типу 'int *', але аргумент 4 має тип 'int8_t * {aka підписаний char *}'. і відскановані значення неправильні. З регулярною змінною int hh не потрібна.
додано Автор Juraj, джерело
AVR має int8_t tm_hour
додано Автор Juraj, джерело
Я використав hh тому, що компілятор попереджав, що це int8_t і це допомогло отримати правильні дані. Я відкрив time.h, але це було для samd або для esp і там був int. У мене було обмежене час, тому я не досліджував його далі. Звичайно,% d призначено для int. Я перевірив int і видалив примітку. Тоді я виявив, що в tm дійсно є unt8_t в тм.
додано Автор Juraj, джерело

Задля повноти тут мій "відповідь" працює з String s, а не char [] .

time_t convertToTime(String calTimestamp) {
  struct tm tm;
  Serial.println("Parsing " + calTimestamp);
  String year = calTimestamp.substring(0, 4);
  String month = calTimestamp.substring(4, 6);
  if (month.startsWith("0")) {
    month = month.substring(1);
  }
  String day = calTimestamp.substring(6, 8);
  if (day.startsWith("0")) {
    month = day.substring(1);
  }
  tm.tm_year = year.toInt() - 1900;
  tm.tm_mon = month.toInt() - 1;
  tm.tm_mday = day.toInt();
  tm.tm_hour = calTimestamp.substring(9, 11).toInt();
  tm.tm_min = calTimestamp.substring(11, 13).toInt();
  tm.tm_sec = calTimestamp.substring(13, 15).toInt();
  return mktime(&tm);
}
1
додано
Цілком ймовірно. Метод String c_str() повертає внутрішній покажчик char * , тому реальне перетворення не потрібне.
додано Автор Sprogz, джерело
Це має спрацювати, але зверніть увагу, що при кожному виклику методу substring() виділяється новий рядок у купі. Навіть якщо у вас є багато вільної пам'яті, численні виклики до malloc() та free() можуть зробити це досить неефективним.
додано Автор Sprogz, джерело
Це бідна бідна купа. Я відчуваю це ...
додано Автор Majenko, джерело
@EdgarBonet так, я знаю, пункт прийнято. У цю функцію подається (частковий) результат відповіді HTTP readStringUntil() . Думаю, було б ефективніше спочатку перетворити calTimestamp на char [] ?
додано Автор Krunal Hingu, джерело