这两个字符串副本 C/C++ 有什么区别
What is the difference between these two String copies C/C++
我写了一个程序来复制strings
。我得到了意想不到的输出。
法典
tper *mem = new tper[MAX_NUM_USERS];
而 tper 看起来像
typedef struct tper {
int8 UserId;
char *b[MAX_NUM_START_LOC];
} tper;
/* PROBLEM */
strncpy(mem->b[num], sourceIds[num]->source, STRING_SIZE);
strcpy(mem->b[num], sourceIds[num]->source);
mem->b[num] = sourceIds[num]->source;
前两个副本在 runtime
时崩溃。第三个副本有效。我不明白为什么?两者都是char *
.
我错过了什么吗?
strcpy
以及strncpy
逐字节将数据从源缓冲区复制到目标缓冲区。因此,首先目标缓冲区大小应该能够容纳所有复制数据。这当然假定两个缓冲区都指向有效的内存区域。其次,源缓冲区必须以 NULL 结尾,除非您希望复制的字节数少于它包含的字节数。请检查您的情况下所有这些是否正常。
可能您的mem->b[num]
没有指向足够大的缓冲区来容纳字符串。
也许mem->b[num]
指针根本没有初始化,或者 NULL 指针。
如果您使用的是 strcpy/strncpy,您有责任确保mem->b[num]
指向一个足够大的有效缓冲区,以包含您要复制到那里的字符串 - 这些函数不会"创建"新字符串,它们只是将字节从一个地方复制到另一个地方,并且由您来确保其他地方是可以存储字节的有效内存。
斯特内皮
strncpy(mem->b[num], sourceIds[num]->source,STRING_SIZE);
这会将所有内容(逐个字节)复制到 sourceIds[num]->source
的 0 字节到mem->b[num]
指向的缓冲区中。
然后,它用 0 字节填充 mem->b[num]
的剩余字节,最多 STRING_SIZE
。
注意:
-
如果
mem->b[num]
实际上没有空间容纳至少STRING_SIZE,strncpy 将在mem->b[num]
末尾复制数据 -
如果
sourceIds[num]->source
正好STRING_SIZE
就没有空间mem->b[num]
中的 0 字节和mem->b[num]
中的数据最终不会成为实际字符串。在这种情况下,您可能需要手动截断字符串,否则将其视为字符串的其他代码可能会超出其边界。例如:mem->b[num][STRING_SIZE - 1] = 0;
结构
strcpy(mem->b[num], sourceIds[num]->source);
这会将所有内容复制到 sourceIds[num]->source
之前,包括从 到 mem->b[num]
的 0 字节。
- 注意:如果
mem->b[num]
实际上没有空间容纳至少strlen(sourceIds[num]->source) + 1
,strcpy 会将数据复制到mem->b[num]
指针赋值
mem->b[num]= sourceIds[num]->source;
这只是将mem->b[num]
分配给指向sourceIds[num]->source
指向的同一事物。不会复制字符串的任何部分。如果更改缓冲区内sourceIds[num]->source
指向的任何内容,则在使用mem->b[num]
时将看到相同的内容,因为它们都指向同一内容。
如果您按顺序执行此操作
tper *mem = new tper[MAX_NUM_USERS];
strncpy(mem->b[num], sourceIds[num]->source, STRING_SIZE);
strcpy(mem->b[num], sourceIds[num]->source);
您的问题是分配mem
不会将数组的任何元素mem->b
设置为指向任何内容。 这些元素都是未初始化的指针,因此访问它们的值(更不用说取消引用它们了,这是strncpy()
和strcpy()
所做的)会产生未定义的行为。
如果为mem->b
元素分配内存以指向 ....
tper *mem = new tper[MAX_NUM_USERS];
for (int i = 0; i < MAX_NUM_START_LOC; ++i)
mem->b[i] = new char[STRING_SIZE];
strncpy(mem->b[num], sourceIds[num]->source, STRING_SIZE);
strcpy(mem->b[num], sourceIds[num]->source);
你会发现它工作,如果sourceIDs[num]
是非 NULL,sourceIds[num]->source
指向一个char
数组的(第一个元素),并且(对于strcpy()
调用)如果strlen(sourceIds[num]->source) < STRING_SIZE
。我假设num
介于 0
和 MAX_NUM_START_LOC-1
之间。
mem->b[num] = sourceIds[num]->source
工作的原因是因为它(大概)是一个简单的指针赋值。 它不会将字符串从 sourceIds[num]->source
复制到 mem->b[num]
。
- C++中std::resize(n)和std::shrink_to_fit之间的区别
- int(c) 和 c-'0' 之间的区别。C++
- 向量 <int> a {N, 0} 和 int arr a[N] = {0} 的时间复杂度有什么区别
- C++ - "!pointer"和"pointer == nullptr"的区别?
- C++ 使用 assign 函数的字符串与直接使用 '=' 更改值的字符串之间的区别
- std::atomic和std::condition_variable wait,notify_*方法之间的区别
- 在 .h 文件中的类中声明静态变量和在.cpp文件中声明"global"变量有什么区别
- 我是C++编程的新手,这些代码之间有什么区别,我应该使用哪一个
- 在 const 函数中通过引用和指针返回之间的区别
- 我想知道长双倍和双倍之间的区别
- 返回常量对象引用 (getter) 和仅返回字符串有什么区别?
- 返回递归调用和仅递归调用的区别
- Qt:remove() 和 rmdir() 有什么区别
- 这 4 个 lambda 表达式之间有什么区别?
- 结构体 S { int align; } 之间的区别;(struct 关键字后的名称)和 struct { int al
- (double) 和 double() 之间的区别
- 将向量作为类>(值)<向量启动和向量<类>[值]有什么区别
- typedef 枚举和枚举类有什么区别?
- &C::c 和 &(C::c) 有什么区别?
- ascii 和 unicode 在处理级别有什么区别吗?