微软的_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;
}