在用sprintf、sprintf_s的过程中遇到的问题

2024-06-23 15:32
文章标签 问题 遇到 过程 sprintf

本文主要是介绍在用sprintf、sprintf_s的过程中遇到的问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

今天在用sprintf和sprintf_s的过程中遇到了问题,想了挺久才解决的,现在就来记录一下。

先上使用sprintf时出现错误的代码

#include <new>
#include <string>int main()
{std::string tmp = "ABC";char *p = new char[tmp.size()]();for (std::size_t index = 0; index < tmp.size(); ++index){sprintf(p + index, "%c", tmp[index]);}delete [] p;return 0;
}

上面的代码会在

delete [] p;

时报告出现越界的错误。

但是按照代码的本意来说,就是应该只把ABC从tmp中移到了p,不会出现越界才对的啊。于是我就在调试时看了一下内存的情况。

 

上面四幅图是对应刚分配内存时、第一次循环时、第二次循环时、第三次循环时的内存状态图。

由第四幅图可以看出,在第三次循环的时候,多了一个0x00,导致越界了,那么这个0x00是从哪里来的呢?看下图

从上图可以看出,"%c"包含了三个字符,%、c、\0,这样的话就能知道越界的原因了,sprintf每次将%c\0复制到了缓冲区里面去了,所以造成最后多了一个0x00,解决方法是只要在分配内存的时候分配多一个字节即可,代码如下

#include <new>
#include <string>int main()
{std::string tmp = "ABC";char *p = new char[tmp.size() + 1]();for (std::size_t index = 0; index < tmp.size(); ++index){sprintf(p + index, "%c", tmp[index]);}delete [] p;return 0;
}

但是上述代码中,如果将sprinf更换为sprintf_s,越界情况还是会出现,那么问题又出现在什么地方了呢?同样道理,先上代码。

#include <new>
#include <string>int main()
{std::string tmp = "ABC";char *p = new char[tmp.size() + 1]();for (std::size_t index = 0; index < tmp.size(); ++index){sprintf_s(p + index, tmp.size() + 1,"%c", tmp[index]);}delete [] p;return 0;
}

上面的代码会在

delete [] p;

时报告出现越界的错误。

于是我又在调试的状态下去看了一下内存的情况。。。

同样,上面四幅图是对应刚分配内存时、第一次循环时、第二次循环时、第三次循环时的内存状态图。

从上述四张图可以看出,sprintf_s函数会在赋值之后将未使用的内存填充为0xfe,而未使用的内存就是由第二个参数减去赋值所占有的内存长度得出的,但是它在计算内存的起点时是由你所填充的内存的位置决定的,于是就出现了上述图片中的现象,41 00 fe fe,41 42 00 fe fe,41 42 43 00 fe fe,造成了越界。那么怎么修改上述代码,修正越界的情况呢?只要修正每次赋值时第二个参数的值即可,代码如下

#include <new>
#include <string>int main()
{std::string tmp = "ABC";char *p = new char[tmp.size() + 1]();for (std::size_t index = 0; index < tmp.size(); ++index){sprintf_s(p + index, tmp.size() + 1 - index,"%c", tmp[index]);}delete [] p;return 0;
}

看来sprintf_s也有很多坑啊。。。

当然也可以用std中的ostringstream来实现sprintf的功能,boost中fmt就更不用说了,但是ostringstream在使用格式化操作的时候真的是。。。很麻烦,所以日常还是使用sprintf会多些。

在这里也贴上一个使用ostringstream格式化操作的代码。代码如下

#include <iostream>
#include <sstream>
#include <iomanip>int main()
{int value = 500;char sz[32] = { 0, };//使用sprintf完成格式化的操作sprintf(sz, "%08X", value);std::cout << "sprintf: " << sz << std::endl;//使用ostringstream完成格式化的操作memset(sz, 0, sizeof(sz));std::ostringstream oss;oss << std::setw(8) << std::setfill('0') << std::hex  << std::setiosflags(std::ios::uppercase) << value;strcpy(sz, oss.str().c_str());std::cout << "ostringstream: " << sz << std::endl;return 0;
}

输出的结果如下

虽然ostringstream比sprintf麻烦,但是ostringstream也比sprintf更安全。毕竟不用自己去管理内存了。

PS:上述全部代码均在Visual Studio 2017上编写。

这篇关于在用sprintf、sprintf_s的过程中遇到的问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1087549

相关文章

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java进程异常故障定位及排查过程

《Java进程异常故障定位及排查过程》:本文主要介绍Java进程异常故障定位及排查过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、故障发现与初步判断1. 监控系统告警2. 日志初步分析二、核心排查工具与步骤1. 进程状态检查2. CPU 飙升问题3. 内存

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

SpringBoot整合liteflow的详细过程

《SpringBoot整合liteflow的详细过程》:本文主要介绍SpringBoot整合liteflow的详细过程,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋...  liteflow 是什么? 能做什么?总之一句话:能帮你规范写代码逻辑 ,编排并解耦业务逻辑,代码

Redis出现中文乱码的问题及解决

《Redis出现中文乱码的问题及解决》:本文主要介绍Redis出现中文乱码的问题及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1. 问题的产生2China编程. 问题的解决redihttp://www.chinasem.cns数据进制问题的解决中文乱码问题解决总结

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示

MySQL中的InnoDB单表访问过程

《MySQL中的InnoDB单表访问过程》:本文主要介绍MySQL中的InnoDB单表访问过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、背景2、环境3、访问类型【1】const【2】ref【3】ref_or_null【4】range【5】index【6】

浏览器插件cursor实现自动注册、续杯的详细过程

《浏览器插件cursor实现自动注册、续杯的详细过程》Cursor简易注册助手脚本通过自动化邮箱填写和验证码获取流程,大大简化了Cursor的注册过程,它不仅提高了注册效率,还通过友好的用户界面和详细... 目录前言功能概述使用方法安装脚本使用流程邮箱输入页面验证码页面实战演示技术实现核心功能实现1. 随机

全面解析MySQL索引长度限制问题与解决方案

《全面解析MySQL索引长度限制问题与解决方案》MySQL对索引长度设限是为了保持高效的数据检索性能,这个限制不是MySQL的缺陷,而是数据库设计中的权衡结果,下面我们就来看看如何解决这一问题吧... 目录引言:为什么会有索引键长度问题?一、问题根源深度解析mysql索引长度限制原理实际场景示例二、五大解决

Springboot如何正确使用AOP问题

《Springboot如何正确使用AOP问题》:本文主要介绍Springboot如何正确使用AOP问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录​一、AOP概念二、切点表达式​execution表达式案例三、AOP通知四、springboot中使用AOP导出