_snprintf_s系列备忘

微软的_snprintf_s总是记不清n的作用。现整理如下。

_snprintf_s有两个版本。一个是带参数模板的,使用于数组,另一个需要指明可操作空间的,适用于堆区变量或数组。

在debug模式下,它还会将'\0'之后至n为止的区域设为-2,以提醒程序员不要把n设定超界。

#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <cassert>

#define LEN 8

int main() {
	const char *srcShort = "01234";
	const char *srcLong = "0123456789";
//数组版本
	char des1[LEN];
	{
		memset(des1, 0, LEN);
		_snprintf_s(des1, LEN - 1, "%s", srcLong);//src若太长则会被截断
		assert(!memcmp(des1, srcLong, LEN - 1) && des1[LEN - 1] == '\0');
	}
	{
		memset(des1, 0, LEN);
		_snprintf_s(des1, strlen(srcShort), "%s", srcShort);//若strlen(srcShort)>LEN-1则会崩溃
		assert(!memcmp(des1, srcShort, strlen(srcShort) + 1) && des1[LEN - 1] == -2);//des1的'\0'后面,都会变为-2.Release版则无此效果
	}
//堆区版本
	char *des2 = static_cast<char*>(malloc(LEN));
	{
		memset(des2, 0, LEN);
		_snprintf_s(des2, LEN, LEN - 1, "%s", srcLong);//src若太长则会被截断
		assert(!memcmp(des2, srcLong, LEN - 1) && des2[LEN - 1] == '\0');
	}
	{
		memset(des2, 0, LEN);
		_snprintf_s(des2, LEN, strlen(srcShort), "%s", srcShort);//若strlen(srcShort)>LEN-1则会崩溃
		assert(!memcmp(des2, srcShort, strlen(srcShort) + 1) && des2[LEN - 1] == -2);//des1的'\0'后面,都会变为-2.Release版则无此效果
	}
	free(des2);
//swprintf_s注意n要设置为字符数而非字节数
	wchar_t *des3 = static_cast<wchar_t*>(malloc(LEN * 2));//字符数为LEN,字节数为LEN*2
	{
		memset(des3, 0, LEN * 2);
		swprintf_s(des3, LEN, L"%ls", L"0");//正确写法,可修改的字符数为LEN(含L'\0'),若超出则崩溃
		free(des3);
	}
	{
		des3 = static_cast<wchar_t*>(malloc(LEN * 2));
		memset(des3, 0, LEN * 2);
		swprintf_s(des3, LEN * 2, L"%ls", L"0");//错误写法,debug版会将des3的L'\0'后的所有字节变为65278,free时会告警
		assert(des3[LEN] == 65278);
		free(des3);//告警了
	}
	return 0;
}

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章