3.13 调用strptime函数前需初始化tm

3.13 调用strptime函数前需初始化tm

2013-11-28 15:26 刘新浙/刘玲/王超/李敬娜 等 人民邮电出版社  字号: T |  T
一键收藏,随时查看,分享好友!

《从缺陷中学习C/C++》第3章库函数问题,本章主要介绍库函数的使用中会遇到的问题。使用库函数可以降低软件开发的难度,提高代码编写的效率。本节为大家介绍调用strptime函数前需初始化tm。

AD:51CTO 网+ 第十二期沙龙:大话数据之美_如何用数据驱动用户体验

3.13  调用strptime函数前需初始化tm

代码示例

   
   
  1. int main()  
  2. {  
  3.     char* format="%Y-%m-%d %H:%M";//没有秒格式符%S  
  4.     char* timeStr="1970-1-1 0:1";//1970年1月1日0时1分,没有秒数  
  5.         struct tm time_info;  
  6.         if(NULL == strptime(timeStr, format, &time_info)) {  
  7.         fprintf(stderr, "error\n");  
  8.     }else{  
  9.         time_t secondsFrom1970 = timegm(&time_info);//这个值期望是60  
  10.         fprintf(stdout, "time =[%s], time in seconds from 1970 is [%d]\n", timeStr, secondsFrom1970);  
  11.     }  
  12.     return 0;  
  13. }  

现象&后果

上述代码运行之后打印出来的并不是期望的60秒。

Bug分析

Linux下man strptime函数可以看到这样一句话,"原则上,这个函数并不会初始化传进来的tm结构,而只是把有指定的字段值存到tm中去。这意味着tm结构需要在调用这个函数之前被初始化"。因此,在调用的strptime时,如果提供的时间字符串不是年月日时分秒都有,并且传递进去的tm结构变量没有被初始化为0,则解析出来的结果是不正确的。

在上述代码中,提供的时间字符串没有秒数,并且没有对tm变量time_info进行初始化,所以运行结果不正确。

正确的做法是,在调用strptime时,要么提供包含"年月日时分秒"的完整时间字符串输入,要么提供的tm结构变量用0初始化。

正确代码

   
   
  1. int main()  
  2. {  
  3.     char* format="%Y-%m-%d %H:%M";//没有秒格式符%S  
  4.     char* timeStr="1970-1-1 0:1";//1970年1月1日0时1分,没有秒数  
  5.         struct tm time_info={0};  
  6.         if(NULL == strptime(timeStr, format, &time_info)) {  
  7.         fprintf(stderr, "error\n");  
  8.         }else{  
  9.         time_t secondsFrom1970 = timegm(&time_info);//这个值期望是60  
  10.         fprintf(stdout, "time =[%s], time in seconds from 1970 is [%d]\n", timeStr, secondsFrom1970);  
  11.     }  
  12.     return 0;  
  13. }  

编程建议

建议在调用strptime函数时,不管提供的时间字符串是否完整,都对tm结构体用0初始化

### C语言 `strptime` 函数使用说明 `strptime` 函数用于将时间字符串按照指定格式解析到 `struct tm` 结构体中。此函数定义如下: ```c char *strptime(const char *buf, const char *format, struct tm *tm); ``` 参数描述: - `const char *buf`: 输入的时间字符串。 - `const char *format`: 解析使用的格式化字符串。 - `struct tm *tm`: 存储解析结果的目标结构。 成功调用后,返回指向未被解析部分的指针;如果输入为空,则返回空指针[^2]。 #### 示例代码展示如何利用 `strptime` 进行基本操作 下面的例子展示了怎样把一个标准形式的时间串转化为内部表示并打印出其组成部分: ```c #include <stdio.h> #include <time.h> int main(void){ // 定义时间和格式化字符串变量 const char* inputTimeStr = "2023-10-09 17:45"; const char* fmt = "%Y-%m-%d %H:%M"; // 创建存储转换后的数据的空间 struct tm parsedTime; // 调用 strptime 将字符串按给定格式解析至 tm 结构体内存区域 if (strptime(inputTimeStr, fmt, &parsedTime)) { printf("Year=%d Month=%d Day=%d Hour=%d Minute=%d\n", parsedTime.tm_year + 1900, parsedTime.tm_mon + 1, parsedTime.tm_mday, parsedTime.tm_hour, parsedTime.tm_min); } else { puts("Failed to parse the string."); } return 0; } ``` 这段程序会尝试读取形如 `"YYYY-MM-DD HH:MM"` 的时间表达式,并将其分解成各个字段输出。注意这里加上的偏移量 (`tm_year + 1900`, `tm_mon + 1`) 是因为这些成员是以不同于常规计数方式初始化的——即年份是从1900年开始计算而月份则是从零开始编号。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值