C ++将长十六进制字符串转换为二进制

Posted

技术标签:

【中文标题】C ++将长十六进制字符串转换为二进制【英文标题】:C++ converting long hex string to binary 【发布时间】:2016-06-02 00:59:54 【问题描述】:

我在这里找到了类似的解决方案: Missing punctuation from C++ hex2bin 并使用该答案:

std::string hex2bin(std::string const& s) 

    std::string sOut;
    sOut.reserve(s.length()/2);

    std::string extract;
    for (std::string::const_iterator pos = s.begin(); pos<s.end(); pos += 2)
    
        extract.assign(pos, pos+2);
        sOut.push_back(std::stoi(extract, nullptr, 16));
    
    return sOut;


int main()

    printf("DECODED: %s\n", hex2bin("5468697320697320612031323320746573742e").c_str());


就像示例中那样打印:

DECODED: This is a 123 test.

但它似乎不适用于较长的十六进制字符串,例如:

printf("DECODED: %s\n", hex2bin("0200000081cd02ab7e569e8bcd9317e2fe99f2de44d49ab2b8851ba4a308000000000000e320b6c2fffc8d750423db8b1eb942ae710e951ed797f7affc8892b0f1fc122bc7f5d74df2b9441a42a14695").c_str());

结果是:

DECODED:

那么如何用 C++ 来完成。

我更熟悉 php,并且有一个函数 hex2bin 可以完成所有工作。

关于此的一点更新: 我正在使用该功能:

std::string hex2bin(std::string s) 
    std::string rc;
    int nLen = s.length();
    int tmp;
    for (int i(0); i + 1 < nLen; i += 2) 
        if (std::istringstream(s.substr(i, 2)) >> std::hex >> tmp) 
            rc.push_back(tmp);
        
    
    return rc;

以及那个函数按字符返回字符: C++

index 0: 2
index 1: 0
index 2: 0
index 3: 0
index 4: -127
index 5: -51
index 6: 2
index 7: -85
index 8: 126
index 9: 86
index 10: -98
index 11: -117
index 12: -51
index 13: -109
index 14: 23
index 15: -30
index 16: -2
index 17: -103
index 18: -14
index 19: -34
index 20: 68
index 21: -44
index 22: -102
index 23: -78
index 24: -72
index 25: -123
index 26: 27
index 27: -92
index 28: -93
index 29: 8
index 30: 0
index 31: 0
index 32: 0
index 33: 0
index 34: 0
index 35: 0
index 36: -29
index 37: 32
index 38: -74
index 39: -62
index 40: -1
index 41: -4
index 42: -115
index 43: 117
index 44: 4
index 45: 35
index 46: -37
index 47: -117
index 48: 30
index 49: -71
index 50: 66
index 51: -82
index 52: 113
index 53: 14
index 54: -107
index 55: 30
index 56: -41
index 57: -105
index 58: -9
index 59: -81
index 60: -4
index 61: -120
index 62: -110
index 63: -80
index 64: -15
index 65: -4
index 66: 18
index 67: 43
index 68: -57
index 69: -11
index 70: -41
index 71: 77
index 72: -14
index 73: -71
index 74: 68
index 75: 26
index 76: 66
index 77: -95
index 78: 70
index 79: -107

PHP

index 0: 2
index 1: 0
index 2: 0
index 3: 0
index 4: 129
index 5: 205
index 6: 2
index 7: 171
index 8: 126
index 9: 86
index 10: 158
index 11: 139
index 12: 205
index 13: 147
index 14: 23
index 15: 226
index 16: 254
index 17: 153
index 18: 242
index 19: 222
index 20: 68
index 21: 212
index 22: 154
index 23: 178
index 24: 184
index 25: 133
index 26: 27
index 27: 164
index 28: 163
index 29: 8
index 30: 0
index 31: 0
index 32: 0
index 33: 0
index 34: 0
index 35: 0
index 36: 227
index 37: 32
index 38: 182
index 39: 194
index 40: 255
index 41: 252
index 42: 141
index 43: 117
index 44: 4
index 45: 35
index 46: 219
index 47: 139
index 48: 30
index 49: 185
index 50: 66
index 51: 174
index 52: 113
index 53: 14
index 54: 149
index 55: 30
index 56: 215
index 57: 151
index 58: 247
index 59: 175
index 60: 252
index 61: 136
index 62: 146
index 63: 176
index 64: 241
index 65: 252
index 66: 18
index 67: 43
index 68: 199
index 69: 245
index 70: 215
index 71: 77
index 72: 242
index 73: 185
index 74: 68
index 75: 26
index 76: 66
index 77: 161
index 78: 70
index 79: 149

所以你可以注意到 C++ 中小于 0 的值在 php 中显示在另一个什么中。我想在 php 中有相同的功能链接。

【问题讨论】:

【参考方案1】:

您的示例包含数字“00”,它将转换为字符值 0,printf 会将其解释为零终止符并停止打印。

【讨论】:

但是为什么呢? php 中的 hex2bin 函数打印一些字符:������~V��͓������DԚ����������������u#ۋ�B�q ח��������+���M��DB�F�。 IMO C++ 函数应该做同样的事情。 @Robert Try C++ std::cout. 那是因为 PHP 是一种不同的语言,具有不同的规则。在 C++ 中,零用作字符串终止符,因此一遇到零就停止打印。【参考方案2】:
printf("DECODED: %s\n", hex2bin("0200000081cd02ab7e569e8bcd9317e2fe99f2de44d49ab2b8851ba4a308000000000000e320b6c2fffc8d750423db8b1eb942ae710e951ed797f7affc8892b0f1fc122bc7f5d74df2b9441a42a14695").c_str());

hex2bin 函数实际上返回一个包含 80 个字符的 std::string。只是显示问题。只需单独打印字符及其整数值即可查看问题:

std::string const s = hex2bin("0200000081cd02ab7e569e8bcd9317e2fe99f2de44d49ab2b8851ba4a308000000000000e320b6c2fffc8d750423db8b1eb942ae710e951ed797f7affc8892b0f1fc122bc7f5d74df2b9441a42a14695");

for (std::string::size_type i = 0; i < s.size(); ++i)

    std::cout << "index " << i << ": " << static_cast<int>(s[i]) << ", printed like this: " << s[i] << "\n";

你会得到这样的输出:

index 0: 2, printed like this: ☻
index 1: 0, printed like this:
index 2: 0, printed like this:

[...]

index 77: -95, printed like this: í
index 78: 70, printed like this: F
index 79: -107, printed like this: ò

在我看来,像hex2bin 这样的函数应该返回std::vector&lt;char&gt;,因为它更好地传达了意图,并且更难打印不正确。不应将二进制数据提供给std::coutprintf,就像它是人类可读的字符串一样,编译器没有任何抱怨的机会。

【讨论】:

a做了一些比较,请阅读我的下一篇文章更清楚

以上是关于C ++将长十六进制字符串转换为二进制的主要内容,如果未能解决你的问题,请参考以下文章

C语言类型转换

C语言itoa()函数和atoi()函数详解(整数转字符C实现)

将长二进制字符串转换为十六进制c#

将长二进制字符串转换为十六进制c#

将长十六进制字符串转换为十进制字符串

C语言itoa()函数和atoi()函数详解(整数转字符C实现)