如何将表示为字符串的整数转换为 ASCII 符号?

How to convert integers represented as string to ASCII symbols?

我有一个输入字符串,看起来像这样:“126022034056098012”。这是我从某个文件中读取的符号的 ASCII 代码串联的结果。例如,代码为 126 22 34 56 98 12。问题是如何将这个字符串解码回字符?注意:字符串不得包含除数字(\、| 等)以外的任何分隔符。接下来我该做什么?

我想出了一种使用 ASCII 符号映射的方法:键-> 带有 ASCII 符号数字表示的字符串,值->ASCII 符号。在循环中,我将传入的数字累积到一个字符串中,直到该字符串与映射中的某个键匹配。匹配时,我将生成的代码转换为字符。我继续,直到我 运行 没有输入数据。但是这种方法适用于字符串和 txt 文件,但不适用于二进制文件。

从 ASCII 码字符串生成字符串的函数:

string Utils::from_number_to_ascii(string number, int size) {
    Utils ut;
    while(number.size() % 3) {
        number = "0" + number;
    }

    string out;
    for (int i = 0; i < size;){
        string st;
        auto it = ut.triple_dict.end();
        while (it == ut.triple_dict.end() && i < size){
            st += number[i++];
            it = ut.triple_dict.find(st);
        }
        out += it->second;
        st = "";
    }
    return out;
}

填充地图:

Utils::Utils() {
    for (int i = 0; i <= 255; i++){
        string s =  to_string(static_cast<int>(i));
        if (s.size() == 1) {
            s = "00" + s;
        } if (s.size() == 2){
            s = "0" + s;
        }
        triple_dict.insert(make_pair(s, static_cast<unsigned char>(i)));
    }
}

不难看出我用三个字节填充了容器:如果symbol的ASCII码是两位数我追加“0”,如果symbol的代码是一位数我追加它用“00”使代码成为三位数。我这样做是为了明确解码符号。

如果每个 ascii 码正好由 3 位数字表示,我们可以很容易地用一个循环来完成:

std::string toAscii(char const* digits, size_t size) {
    std::string output(size / 3, '[=10=]');
    for(char& c : output) {
        char d0 = *digits++; // Get 3 digits
        char d1 = *digits++;
        char d2 = *digits++; 

        int ascii_value = (d0 - '0') * 100 + (d1 - '0') * 10 + (d2 - '0'); 
        c = (char)ascii_value; 
    }
    return output; 
}

用法示例

我有一个带有示例输入的 c-string,以及一个带有预期输出的字符串。该程序验证它们是否相等。

int main() {
    auto&& input = "126022034056098012";
    std::string expected_output = {char(126), char(22), char(34), char(56), char(98), char(12)};
    std::cout << (toAscii(input, sizeof(input)) == expected_output); // Prints true
}

fstream.write() 是否在文件末尾添加'\0'?

否。 如果您的字符串包含 0 字符,它会添加它,否则不会。我们可以使用一些非常短的示例代码来自行测试。

#include <fstream>
#include <iostream>
#include <string>

int main()
{
    {
        std::ofstream file("test.txt");
        std::string   message = "Hello!";

        file.write(message.data(), message.length());
        // file gets closed automatically 
    }
    {
        std::ifstream file("test.txt");


        while (file)
        {
            std::cout << file.get() << '\n';
        }
        // file gets closed automatically
    }
}

当我编译并 运行 这段代码时,它输出以下内容。每个值都对应"Hello!"中对应字符的值,最后一个除外。 -1 表示您已到达文件末尾,但如果您使用的是 file.read 之类的方法,它就不会出现。 [=17=] 没有出现在文件中的任何位置。

72
101
0
108
111
33
-1