const char * 和 std::string.c_str()引起的坑

本文详细解析了在编码转换过程中,使用constchar*和std::string.c_str()引发的潜在问题。通过具体案例,展示了不当使用c_str()可能导致的指针失效及数据丢失风险,并提供了解决方案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

const char *和std::string.c_str引起的坑

今天早上有同事说调用封装好的编码转换函数,有时候能转换成功,有时候转换会失败,一听现象我就猜测是用法导致的问题,定位下来果然是用法使用有误,当然也跟我们封装的函数有关,记录下来,避免再次采坑。

一,发现问题

该编码转换函数实现如下:

//ANSI转UTF8
static std::string AnsiToUtf8(const std::string &strAnsi)
{
	std::wstring strUnicode = AnsiToUnicode(strAnsi);
	return UnicodeToUTF8(strUnicode);
}


//UNICODE转UTF8
static std::string UnicodeToUTF8(const std::wstring& strUnicode)
{
	int nUtf8Length = WideCharToMultiByte(CP_UTF8,
		0,
		strUnicode.c_str(),
		-1,
		NULL,
		0,
		NULL,
		NULL);

	char* pUtf8 = new char[nUtf8Length + 1];
	memset((void*)pUtf8, 0, sizeof(char) * (nUtf8Length + 1));

	::WideCharToMultiByte(CP_UTF8,
		0,
		strUnicode.c_str(),
		-1,
		pUtf8,
		nUtf8Length,
		NULL,
		NULL);

	std::string strUtf8;
	strUtf8 = pUtf8;
	delete[] pUtf8;

	return strUtf8;
}

实现ANSI编码转换为utf8编码,函数返回值是一个局部变量std::string,该同事在外部调用的时候直接用const char* p = UnicodeToUTF8().c_str()来接收,可是c_str()就只是返回一个地址而已,而这个地址可能会随着string 对象的销毁(比如此处的局部对象)而变得无效!!
调用处的代码大概如下所示:

const char *p1 = AnsiToUtf8("你好").c_str();
std::string str3 = Utf8ToAnsi(p1);

此时调试发现这个指针p1的地址指向此时是一个无效值,同时调用Utf8ToAnsi将该utf8编码的字符串转换回来会发现转换失败。
在这里插入图片描述而采用

std::string str1 = AnsiToUtf8("你好");

写法程序能够正常运行,这是因为使用了一个新的局部变量std::string str1来接收了函数的返回值,str1内部维护了一个新的内存。

二,解决方法

当初实现编码转换函数的时候,本意也是使用c++语言来调用的,使用std::string来接收返回值的,加上没有想到std::string的c_str()函数这个坑,导致此问题的出现。由于其他处的代码使用没有问题,暂时就修改该处使用不当的代码,并且在编码转换接口上写上备注,避免其他开发人员犯同样的错误。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Simple Simple

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值