23

我需要一个如何使用 Crypto++ 从 std::string 生成 SHA256 哈希并输出 std::string 的示例。我似乎无法弄清楚。我尝试过的一切都给了我无效的输出。

这是interjay回答后的新代码:

string SHA256(string data)
{
    byte const* pbData = (byte*) data.data();
    unsigned int nDataLen = data.size();
    byte abDigest[CryptoPP::SHA256::DIGESTSIZE];

    CryptoPP::SHA256().CalculateDigest(abDigest, pbData, nDataLen);

    return string((char*)abDigest);
}

SHA256("A") 的输出;是

在此处输入图像描述

我怎样才能把它变成可读的格式?

感谢 interjay 的回答,我能够生成最终的哈希值。

4

3 回答 3

24

这使用以下输出 base64 字符串CryptoPP::Base64Encoder

#include "sha.h"
#include "filters.h"
#include "base64.h"

std::string SHA256HashString(std::string aString){
    std::string digest;
    CryptoPP::SHA256 hash;

    CryptoPP::StringSource foo(aString, true,
    new CryptoPP::HashFilter(hash,
      new CryptoPP::Base64Encoder (
         new CryptoPP::StringSink(digest))));

    return digest;
}
于 2011-08-12T20:16:27.960 回答
20

此行将给出错误的结果:

unsigned int nDataLen = sizeof(pbData);

它总是会给你一个指针的大小。你想要的是data.size().

此外,您不需要这部分:

if(!CryptoPP::SHA256().VerifyDigest(abDigest, pbData, nDataLen))
{
    return SHA256(data);
}

它应该始终正确验证,因为您只是根据相同的数据计算了摘要。如果没有,您将进入无限递归。

要获得可读的输出,您可以将其转换为十六进制。这是Crypto++ Wiki中的 MD5 示例,如果您将 MD5 替换为 SHA256,它应该对您有用:

CryptoPP::MD5 hash;
byte digest[ CryptoPP::MD5::DIGESTSIZE ];
std::string message = "abcdefghijklmnopqrstuvwxyz";

hash.CalculateDigest( digest, (byte*) message.c_str(), message.length() );

CryptoPP::HexEncoder encoder;
std::string output;
encoder.Attach( new CryptoPP::StringSink( output ) );
encoder.Put( digest, sizeof(digest) );
encoder.MessageEnd();

std::cout << output << std::endl;  
于 2011-05-08T20:35:06.943 回答
1

您的代码将期望您提供给字符串构造函数的缓冲区中有一个以空字符结尾的字符串!这意味着结果几乎肯定是错误的。

要强制执行摘要大小,请改用以下内容:

return std::string((char*)abDigest, CryptoPP::SHA256::DIGESTSIZE);

同样关于打印它,以下正确生成BA7816BF8F01CFEA414140DE5DAE2223B00361A396177A9CB410FF61F20015AD字符串的测试向量"abc"

std::string string_to_hex(const std::string& input)
{
  static const char* const lut = "0123456789ABCDEF";
  size_t len = input.length();

  std::string output;
  output.reserve(2 * len);
  for (size_t i = 0; i < len; ++i)
  {
    const unsigned char c = input[i];
    output.push_back(lut[c >> 4]);
    output.push_back(lut[c & 15]);
  }
  return output;
}

std::string SHA256(std::string data)
{
  CryptoPP::byte const* pbData = (CryptoPP::byte*)data.data();
  unsigned int nDataLen = data.length();
  CryptoPP::byte abDigest[CryptoPP::SHA256::DIGESTSIZE];

  CryptoPP::SHA256().CalculateDigest(abDigest, pbData, nDataLen);

  // return string((char*)abDigest);  -- BAD!!!
  return std::string((char*)abDigest, CryptoPP::SHA256::DIGESTSIZE);
}

void test_cryptopp() {
  std::cout << string_to_hex(SHA256("abc")) << std::endl;
}
于 2018-02-22T22:27:38.460 回答