1

在仔细查看 time.h 多次后,我编写了以下 func :

void output_date ( int day, int month, int year ) {
    char buffer[64] = "";
    struct tm *e_time = calloc( (size_t) 1, sizeof(struct tm) );

    e_time->tm_year = year - 1900;
    e_time->tm_mon = month - 1;
    e_time->tm_mday = day;
    e_time->tm_hour = 0;
    e_time->tm_min = 0;
    e_time->tm_sec = 0;
    e_time->tm_isdst = -1;

    /* strftime ( buffer, 64, (char *)0, e_time ); */
    strftime ( buffer, 64, "%a %b %e %H:%M:%S %Z (%z) %Y", e_time );

    printf ( "%s\n", buffer );

    free(e_time);
    e_time = NULL;

}

然后我用一系列可能的输入调用了这个函数,除了奇怪的输出之外什么也没看到:

Sun Jul  8 00:00:00  () 2013
Sun Jul  9 00:00:00  () 2013
Sun Jul 10 00:00:00  () 2013
Sun Jul 11 00:00:00  () 2013
Sun Jul 12 00:00:00  () 2013

当我将格式字符串测试为 strftime 时,我看到了很好的结果:

$ date -u "+%a %b %e %H:%M:%S %Z (%z) %Y"
Sat Oct 12 00:40:05 GMT (+0000) 2013

我什至通过调试器单步执行并看到了同样奇怪的结果:

stopped in main at line 27 in file "flight.c"
   27                       output_date ( day+1, month+1, year );
(dbx) print day+1, month+1, year
day+1 = 1
month+1 = 1
year = 1977

(dbx) step                      
stopped in output_date at line 43 in file "flight.c"
   43       char buffer[64] = "";
(dbx) step
stopped in output_date at line 44 in file "flight.c"
   44       struct tm *e_time = calloc( (size_t) 1, sizeof(struct tm) );
(dbx) step
stopped in output_date at line 46 in file "flight.c"
   46       e_time->tm_year = year - 1900;
(dbx) print e_time
e_time = 0x100101640
(dbx) step        
stopped in output_date at line 47 in file "flight.c"
   47       e_time->tm_mon = month - 1;
(dbx) step
stopped in output_date at line 48 in file "flight.c"
   48       e_time->tm_mday = day;
(dbx) step
stopped in output_date at line 49 in file "flight.c"
   49       e_time->tm_hour = 0;
(dbx) step
stopped in output_date at line 50 in file "flight.c"
   50       e_time->tm_min = 0;
(dbx) step
stopped in output_date at line 51 in file "flight.c"
   51       e_time->tm_sec = 0;
(dbx) step
stopped in output_date at line 52 in file "flight.c"
   52       e_time->tm_isdst = -1;
(dbx) step
stopped in output_date at line 55 in file "flight.c"
   55       strftime ( buffer, 64, "%a %b %e %H:%M:%S %Z (%z) %Y", e_time );
(dbx) print *e_time
*e_time = {
    tm_sec   = 0
    tm_min   = 0
    tm_hour  = 0
    tm_mday  = 1
    tm_mon   = 0
    tm_year  = 77
    tm_wday  = 0
    tm_yday  = 0
    tm_isdst = -1
}
(dbx) step        
stopped in output_date at line 57 in file "flight.c"
   57       printf ( "%s\n", buffer );
(dbx) print buffer
buffer = "Sun Jan  1 00:00:00  () 1977"
(dbx) quit    

事实上,我所得到的只是一周中的某一天是星期天,以及正确的月份和正确的日期和年份。似乎没有什么是正确的。

我错过了一些明显的东西吗?

4

2 回答 2

1

您应该mktime(e_time)在填充结构之后调用,但在调用strftime(). 您还有一些其他结构成员当前尚未完成。传递具有未初始化值的结构(或在此处初始化为不正确的值,由于您的calloc()调用)通常是一件坏事,并mktime()修改您传递的结构以填充它们。

像这样:

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

void output_date ( int day, int month, int year ) {
    char buffer[64] = "";
    struct tm *e_time = calloc( (size_t) 1, sizeof(struct tm) );

    e_time->tm_year = year - 1900;
    e_time->tm_mon = month - 1;
    e_time->tm_mday = day;
    e_time->tm_hour = 0;
    e_time->tm_min = 0;
    e_time->tm_sec = 0;
    e_time->tm_isdst = -1;

    if ( mktime(e_time) < 0 ) {
        fprintf(stderr, "Error getting calendar time.\n");
        exit(EXIT_FAILURE);
    }

    int n = strftime ( buffer, 64, "%a %b %e %H:%M:%S %Z (%z) %Y", e_time );
    printf("Return from strftime() is %d\n", n);

    printf ( "%s\n", buffer );

    free(e_time);
    e_time = NULL;

}

int main(void) {
    output_date(12, 6, 2013);
    return 0;
}

产量:

paul@local:~/src/c/scratch$ ./st
Return from strftime() is 36
Wed Jun 12 00:00:00 EDT (-0400) 2013
paul@local:~/src/c/scratch$
于 2013-10-12T01:00:27.303 回答
1

您为该字段tm_isdst指定了 的值-1,这意味着夏令时不可用,1请将其更改为夏令时是否有效0

正如@Paul Griffiths 指出的那样,调用mktime(e_time)是这里更好的选择,该mktime函数将填充分解的e_time,因为它也将修复文件,如tm_wdaytm_day。对于 的字段tm_isdst,它遵循以下规则:

的正值或零值tm_isdst使mktime函数最初分别假定夏令时在指定时间内有效或无效。负值会导致它尝试确定夏令时是否在指定时间内有效。

于 2013-10-12T01:03:11.340 回答