起因:操作系统实验要求写一个文件管理系统,要求输出一个文件的创建时间,修改时间和上次访问时间。在文件的结构体中创建了三个time_t类型变量用于保存时间的值,需要输出时调用ctime()函数格式化输出,遇到了一些问题记录一下。
创新互联建站主营平陆网站建设的网络公司,主营网站建设方案,重庆App定制开发,平陆h5小程序定制开发搭建,平陆网站营销推广欢迎平陆等地区企业咨询问题一:调用ctime()后结尾自动换行。
问题复现:
time_t ptime;
time(&ptime);
char *s = ctime(&ptime);
printf("%s\n", s);
输出结果:
最后一行多了一个换行符,保险起见讲字符串每个字符转为成ASCII码打印出来:
time_t ptime;
time(&ptime);
char *s = ctime(&ptime);
// printf("%s\n", s);
for (int i = 0; i< strlen(s); i++)
{
printf("%d ", *(s + i));
}
printf("\n");
结尾是10,多了一个换行符无疑,这个问题倒是好解决:
time_t ptime;
time(&ptime);
char *s = ctime(&ptime);
*(s + strlen(s) - 1) = 0;
printf("%s\n", s);
再次输出:
结尾的换行符去掉了,问题解决
问题二:问题复现:
char *dt1, *dt2, *dt3;
time_t ptime;
time(&ptime);
dt1 = ctime(&ptime);
dt2 = ctime(&ptime);
dt3 = ctime(&ptime);
*(dt1 + strlen(dt1) - 1) = 0;
*(dt2 + strlen(dt2) - 1) = 0;
*(dt3 + strlen(dt3) - 1) = 0;
printf("%s\n%s\n%s\n", dt1, dt2, dt3);
结果:
不仅最后的换行符没了,连标准化输出的最后几个字符也没了。
于是用gcc一调试:
三个指针指向了同一个地址,所以用上述方法去掉结尾换行符的时候相当于把最后3个字节全改成了0,2022\n变成了20\0\0\0。
明明调用了三次ctime(),但是每一次指向的是同一个地址,很迷,然后继续调,看ctime()函数的实现:
static __inline char *__CRTDECL ctime(const time_t *_Time) { return _ctime64(_Time); }
返回值是是一个指针,这个_ctime64()不知道是什么,上微软官网一查:
看到了这么一段话,总结就是ctime()和gmtime(),localtime()共享一个静态缓存区,每一次调用都会覆盖掉之前的值。那也就是说,在代码中的三个指针指向的是同一个地址,每次修改都是对同一片内存进行修改,修改了三次,所以才会导致输出异常。
解决方法:
char *dt1, *dt2, *dt3;
time_t ptime;
time(&ptime);
dt1 = (char *)malloc(25 * sizeof(char));
dt2 = (char *)malloc(25 * sizeof(char));
dt3 = (char *)malloc(25 * sizeof(char));
strcpy(dt1, ctime(&ptime));
strcpy(dt2, ctime(&ptime));
strcpy(dt3, ctime(&ptime));
*(dt1 + strlen(dt1) - 1) = 0;
*(dt2 + strlen(dt2) - 1) = 0;
*(dt3 + strlen(dt3) - 1) = 0;
printf("%s\n%s\n%s\n", dt1, dt2, dt3);
给三个指针分配一个自己的空间然后把值复制过去,问题就解决了:
输出正常。
你是否还在寻找稳定的海外服务器提供商?创新互联www.cdcxhl.cn海外机房具备T级流量清洗系统配攻击溯源,准确流量调度确保服务器高可用性,企业级服务器适合批量采购,新人活动首月15元起,快前往官网查看详情吧