asctime - день месяца ноль или пробел? - PullRequest
0 голосов
/ 22 декабря 2018

У меня есть следующая программа, демонстрирующая использование asctime.

#include <stdio.h>
#include <time.h>

int main(void) {
    struct tm   broken_down;
    broken_down.tm_year = 2000 - 1900;
    broken_down.tm_mon = 0;
    broken_down.tm_mday = 1;
    broken_down.tm_hour = broken_down.tm_min = broken_down.tm_sec = 0;

    printf("Current date and time: %s", asctime(&broken_down));
}

Эта программа печатает Current date and time: Sun Jan 1 00:00:00 2000 на ideone.com , то есть поле даты является пробеломpadded.

Когда я компилирую и запускаю эту программу с MSVC, она создает строку даты с нулем в начале месяца: Current date and time: Sun Jan 01 00:00:00 2000.

В чем причина этого несоответствия?Какой формат правильный?

1 Ответ

0 голосов
/ 22 декабря 2018

Как обычно, авторы (не) стандартной библиотеки C от Microsoft не задумывались о правильной реализации буквы стандарта.

Даже в оригинальном стандарте C89 / C90 появляется следующий текст

Описание

Функция asctime преобразует время разбивки в структуре, на которую указывает timeptr, в строку вформа

Sun Sep 16 01:03:52 1973\n\0

с использованием эквивалента следующего алгоритма.

char *asctime(const struct tm *timeptr)
{
    static const char wday_name[7][3] = {
             "Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"
    };
    static const char mon_name[12][3] = {
             "Jan", "Feb", "Mar", "Apr", "May", "Jun",
             "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"
    };
    static char result[26];

    sprintf(result, "%.3s %.3s%3d %.2d:%.2d:%.2d %d\n",
             wday_name[timeptr->tm_wday],
             mon_name[timeptr->tm_mon],
             timeptr->tm_mday, timeptr->tm_hour,
             timeptr->tm_min, timeptr->tm_sec,
             1900 + timeptr->tm_year);
    return result;
}

Сам пример, к сожалению, использует дату, которая имеет двухзначный день месяца, но код использует %3d, что означает десятичную цифру с пробелами и выровненные по правому краюв пределах 3-символьного поля .

Результат для данного разбитого времени равен Sun Jan 1 00:00:00 2000 с пробелом.


Python 2, до тех пор, пока 2.7.15 не выставлял стандартную библиотеку C * Вывод 1032 * как есть, за исключением новой строки, которая вызывала платформо-зависимое поведение, теперь в 2.7.15 исправлено использование жестко закодированногоформат с ведущим пробелом.Документация Python 2 также использует дату с двузначным числом месяца в своем примере, что добавляет путаницу.

...