余编码下面的函数转换一个std::vector的uint8_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函数.有什么建议吗?
(注意:答案是针对标准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)
您可以直接使用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)
| 归档时间: |
|
| 查看次数: |
2541 次 |
| 最近记录: |