4

这是我从未擅长的 C/C++ 领域之一。

我的问题是我有一个最终需要包含一些空字符的字符串。将所有内容都视为 char 数组(或字符串)是行不通的,因为当他们找到第一个 null 时,事情往往会崩溃。所以我想,好吧,我会切换到 uint8_t,所以一切都只是一个数字。我可以根据需要移动东西,并在我准备好时将其转换回字符。

我现在的主要问题是:如何将字符串的一部分复制到 uint8_t 缓冲区?

实际上,我想做类似的事情:

std::string s = "abcdefghi";
uint8_t *val = (uint8_t*)malloc(s.length() + 1);
memset(val, 0, s.length() + 1);

// Assume offset is just some number
memcpy(val + offset, s.substr(1, 5).c_str(), 5);

显然,当我尝试这个时,我得到了一个错误。在 memcpy 的第一个参数中可能存在某种诡计(我在网上看到 (*(uint8_t*)) 之类的东西,但不知道这意味着什么)。

有什么帮助吗?

当我在这里时,我怎样才能轻松地将其转换回 char 数组?只是将 uint8_t 指针静态转换为 char 指针?

多谢。

4

2 回答 2

1

我想,好吧,我会切换到 uint8_t,所以一切都只是一个数字。

这不会使寻找'\0'的算法突然停止这样做,使用char的算法也不必注意'\0'。用空字符表示结束是 C 字符串的约定,而不是 char 数组。uint8_t 可能只是 char 的 typedef。

正如 Nicol Bolas 指出的那样,std::string 已经能够存储包含空字符的字符串,而无需特殊处理空字符。

至于你的问题,我不确定你指的是什么错误,因为以下工作正常:

#include <iostream>
#include <string>
#include <cstdint>
#include <cstring>

int main() {
    std::string s = "abcdefghi";
    std::uint8_t *val = (std::uint8_t*)std::malloc(s.length() + 1);
    std::memset(val, 0, s.length() + 1);

    int offset = 2;
    std::memcpy(val + offset, s.substr(1, 5).c_str(), 5);
    std::cout << (val+offset) << '\n';
}

memcpy 行从字符串中取出第二个到第六个字符s并将它们复制到 val 中。然后带有 cout 的行打印“bcdef”。

当然这是 C++,所以如果你想手动分配一些内存并将其归零,你可以这样做:

std::unique_ptr<uint8_t[]> val(new uint8_t[s.length()+1]());

或使用向量:

std::vector<uint8_t> val(s.length()+1,0);

要从 uint8_t 数组进行转换,您可以(但通常不应该)执行以下操作:

char *c = reinterpret_cast<uint8_t*>(val);
于 2012-05-31T20:18:09.773 回答
0

好吧,代码工作正常,它复制了 val 中的子字符串。但是,在偏移之前,您将在所有位置上都有 0。

例如对于 offset=2 val 将是 {0, 0, b, c, d, e, f, 0, 0, 0}

如果你打印它,它不会显示任何内容,因为字符串在第一个位置以空结尾(我想这是你正在谈论的错误......)。

于 2012-05-31T19:59:30.817 回答