将uint8的向量转换为ascii十六进制字符串的更好方法

Fil*_*ria 1 c++ vector std

余编码下面的函数转换一个std::vectoruint8_t一个ASCII十六进制string (GNU ++ 98标准).

 ...
string uint8_vector_to_hex_string(const vector<uint8_t>& v) {
    stringstream ss;
    vector<uint8_t>::const_iterator it;

    for (it = v.begin(); it != v.end(); it++) {
        char hex_char[2];
        sprintf(hex_char, "%x", *it);
        ss << "\\x" << hex_char;
    }

    return ss.str();
}
 ...
Run Code Online (Sandbox Code Playgroud)

它工作正常.我想知道是否有更好的方法来进行这种转换,也许不使用stringstream对象和sprintf函数.有什么建议吗?

Ker*_* SB 5

(注意:答案是针对标准C++;对于较旧的方言,可能需要对范围迭代进行微小的修改.我认为对于算法效率的核心问题而言,这些问题非常重要.)


通过意识到你基本上知道答案并且不需要完成所有这些动态工作,你可以节省大量的工作.

std::string uint8_vector_to_hex_string(const vector<uint8_t>& v)
{
    std::string result;
    result.reserve(v.size() * 2);   // two digits per character

    static constexpr char hex[] = "0123456789ABCDEF";

    for (uint8_t c : v)
    {
        result.push_back(hex[c / 16]);
        result.push_back(hex[c % 16]);
    }

    return result;
}
Run Code Online (Sandbox Code Playgroud)

本着"识别算法"的精神,这里是一个用于数值序列的位值格式化的分离算法.用例:

#include <iostream>
#include <string>
#include <vector>

// bring your own alphabet
constexpr char Alphabet[] = "0123456789ABCDEF";

// input
std::vector<unsigned char> const v { 31, 214, 63, 9 };

// output (Note: *our* responsibility to make allocations efficient)
std::string out;
out.reserve(v.size() * 2);

// the algorithm
place_value_format<char,     // output type
                      2,     // fixed output width
                     16>(    // place-value number base
    v.begin(), v.end(),      // input range
    std::back_inserter(out), // output iterator
    Alphabet);               // digit representation
Run Code Online (Sandbox Code Playgroud)

现在算法:

#include <algorithm>
#include <iterator>

template <typename Out, std::size_t NDigits, std::size_t Base,
          typename InItr, typename OutItr>
OutItr place_value_format(InItr first, InItr last, OutItr out, Out const * digits)
{
    for (; first != last; ++first)
    {
        Out unit[NDigits];

        auto val = *first;

        for (auto it = std::rbegin(unit); it != std::rend(unit); ++it)
        {
            *it = digits[val % Base];
            val /= Base;
        }

        out = std::copy(std::begin(unit), std::end(unit), out);
    }

    return out;
}
Run Code Online (Sandbox Code Playgroud)


Tub*_*iar 5

您可以直接使用stringstream进行十六进制格式化:

...
string uint8_vector_to_hex_string(const vector<uint8_t>& v) {
    stringstream ss;
    ss << std::hex << std::setfill('0');
    vector<uint8_t>::const_iterator it;

    for (it = v.begin(); it != v.end(); it++) {
        ss << "\\x" << std::setw(2) << static_cast<unsigned>(*it);
    }

    return ss.str();
}
...
Run Code Online (Sandbox Code Playgroud)