一、SHA256算法介绍与原理详解
见博客《SHA256算法原理详解》
二、SHA256算法C++实现
1)sha256.hpp
/*
* Filename: sha256.hpp
* Author: L.Y.
* Brief: SHA256算法实现
* Version: V1.0.0
* Update log:
* 1)20191108-20191113 V1.0.0
* 1、初次版本。
* TODO:
* Attention:
* 1)输入信息中有中文时,得到的数字指纹与使用SHA256在线加密工具得到数字指纹可能不相同。
* 原因是中文的编码方式不同。
*/
#ifndef SHA256_HPP
#define SHA256_HPP
#include
#include
#include
namespace ly
{
//
// \brief: SHA256算法实现
//
class Sha256
{
public:
//! 默认构造函数
Sha256() {}
//! 析构函数
virtual ~Sha256() {}
/** @brief: 使用SHA256算法,获取输入信息的摘要(数字指纹)
@param[in] message: 输入信息
@param[out] _digest: 摘要(数字指纹)
@return: 是否成功
*/
bool encrypt(const std::vector& message,
std::vector* _digest);
/** @brief: 获取十六进制表示的信息摘要(数字指纹)
@param[in] message: 输入信息
@return: 十六进制表示的信息摘要(数字指纹)
*/
std::string getHexMessageDigest(const std::string& message);
protected:
/// SHA256算法中定义的6种逻辑运算 ///
inline uint32_t ch(uint32_t x, uint32_t y, uint32_t z) const;
inline uint32_t maj(uint32_t x, uint32_t y, uint32_t z) const;
inline uint32_t big_sigma0(uint32_t x) const;
inline uint32_t big_sigma1(uint32_t x) const;
inline uint32_t small_sigma0(uint32_t x) const;
inline uint32_t small_sigma1(uint32_t x) const;
/** @brief: SHA256算法对输入信息的预处理,包括“附加填充比特”和“附加长度值”
附加填充比特: 在报文末尾进行填充,先补第一个比特为1,然后都补0,直到长度满足对512取模后余数是448。需要注意的是,信息必须进行填充。
附加长度值: 用一个64位的数据来表示原始消息(填充前的消息)的长度,并将其补到已经进行了填充操作的消息后面。
@param[in][out] _message: 待处理的信息
@return: 是否成功
*/
bool preprocessing(std::vector* _message) const;
/** @brief: 将信息分解成连续的64Byte大小的数据块
@param[in] message: 输入信息,长度为64Byte的倍数
@param[out] _chunks: 输出数据块
@return: 是否成功
*/
bool breakTextInto64ByteChunks(const std::vector& message,
std::vector>* _chunks) const;
/** @brief: 由64Byte大小的数据块,构造出64个4Byte大小的字。
构造算法:前16个字直接由数据块分解得到,其余的字由如下迭代公式得到:
W[t] = small_sigma1(W[t-2]) + W[t-7] + small_sigma0(W[t-15]) + W[t-16]
@param[in] chunk: 输入数据块,大小为64Byte
@param[out] _words: 输出字
@return: 是否成功
*/
bool structureWords(const std::vector& chunk,
std::vector* _words) const;
/** @breif: 基于64个4Byte大小的字,进行64次循环加密
@param[in] words: 64个4Byte大小的字
@param[in][out] _message_digest: 信息摘要
@return: 是否成功
*/
bool transform(const std::vector& words,
std::vector* _message_digest) const;
/** @brief: 输出最终的哈希值(数字指纹)
@param[in] input: 步长为32bit的哈希值
@param[out] _output: 步长为8bit的哈希值
@return: 是否成功
*/
bool produceFinalHashValue(const std::vector& input,
std::vector* _output) const;
private:
static std::vector initial_message_digest_; // 在SHA256算法中的初始信息摘要,这些常量是对自然数中前8个质数的平方根的小数部分取前32bit而来。
static std::vector add_constant_; // 在SHA256算法中,用到64个常量,这些常量是对自然数中前64个质数的立方根的小数部分取前32bit而来。
};
/ 内联函数&模版函数的定义 /
inline uint32_t Sha256::ch(uint32_t x, uint32_t y, uint32_t z) const
{
return (x & y) ^ ((~x) & z);
}
inline uint32_t Sha256::maj(uint32_t x, uint32_t y, uint32_t z) const
{
return (x & y) ^ (x & z) ^ (y & z);
}
inline uint32_t Sha256::big_sigma0(uint32_t x) const
{
return (x >> 2 | x << 30) ^ (x >> 13 | x << 19) ^ (x >> 22 | x << 10);
}
inline uint32_t Sha256::big_sigma1(uint32_t x) const
{
return (x >> 6 | x << 26) ^ (x >> 11 | x << 21) ^ (x >> 25 | x << 7);
}
inline uint32_t Sha256::small_sigma0(uint32_t x) const
{
return (x >> 7 | x << 25) ^ (x >> 18 | x << 14) ^ (x >> 3);
}
inline uint32_t Sha256::small_sigma1(uint32_t x) const
{
return (x >> 17 | x << 15) ^ (x >> 19 | x << 13) ^ (x >> 10);
}
} // namespace ly
#endif // SHA256_HPP
2)sha256.cpp
/*
* Filename: sha256.cpp
* Author: L.Y.
* Brief: SHA256算法实现
* Version: V1.0.0
* Update log:
* 1)20191108-20191113 V1.0.0
* 1、初次版本。
* TODO:
* Attention:
* 1)输入信息中有中文时,得到的数字指纹与使用SHA256在线加密工具得到数字指纹可能不相同。
* 原因是中文的编码方式不同。
*/
#include "sha256.hpp"
#include
#include
#include
#include
namespace ly
{
静态数据成员初始化 //
std::vector Sha256::initial_message_digest_ =
{
0x6a09e667, 0xbb67ae85, 0x3c6ef372,
0xa54ff53a, 0x510e527f, 0x9b05688c,
0x1f83d9ab, 0x5be0cd19
};
std::vector Sha256::add_constant_ =
{
0x428a2f98,0x71374491,0xb5c0fbcf,0xe9b5dba5,0x3956c25b,0x59f111f1,0x923f82a4,0xab1c5ed5,
0xd807aa98,0x12835b01,0x243185be,0x550c7dc3,0x72be5d74,0x80deb1fe,0x9bdc06a7,0xc19bf174,
0xe49b69c1,0xefbe4786,0x0fc19dc6,0x240ca1cc,0x2de92c6f,0x4a7484aa,0x5cb0a9dc,0x76f988da,
0x983e5152,0xa831c66d,0xb00327c8,0xbf597fc7,0xc6e00bf3,0xd5a79147,0x06ca6351,0x14292967,
0x27b70a85,0x2e1b2138,0x4d2c6dfc,0x53380d13,0x650a7354,0x766a0abb,0x81c2c92e,0x92722c85,
0xa2bfe8a1,0xa81a664b,0xc24b8b70,0xc76c51a3,0xd192e819,0xd6990624,0xf40e3585,0x106aa070,
0x19a4c116,0x1e376c08,0x2748774c,0x34b0bcb5,0x391c0cb3,0x4ed8aa4a,0x5b9cca4f,0x682e6ff3,
0x748f82ee,0x78a5636f,0x84c87814,0x8cc70208,0x90befffa,0xa4506ceb,0xbef9a3f7,0xc67178f2
};
成员函数的定义 //
bool Sha256::encrypt(const std::vector& input_message,
std::vector* _digest)
{
if (!input_message.empty() && _digest)
{
//! 文本预处理
std::vector message = input_message;
preprocessing(&message);
//! 将文本分解成连续的64Byte大小的数据块
std::vector> chunks;
breakTextInto64ByteChunks(message, &chunks);
//! 由64Byte大小的数据块,构造出64个4Byte大小的字。然后进行循环迭代。
std::vector message_digest(initial_message_digest_); // 初始化信息摘要
std::vector words;
for (const auto& chunk : chunks)
{
structureWords(chunk, &words);
transform(words, &message_digest);
}
//! 获取最终结果
produceFinalHashValue(message_digest, _digest);
return true;
}
else
{
return false;
}
}
std::string Sha256::getHexMessageDigest(const std::string& message)
{
if (!message.empty())
{
std::vector __message;
for (auto it = message.begin(); it != message.end(); ++it)
{
__message.push_back(static_cast(*it));
}
std::vector digest;
encrypt(__message, &digest);
std::ostringstream o_s;
o_s << std::hex << std::setiosflags(std::ios::uppercase);
for (auto it = digest.begin(); it != digest.end(); ++it)
{
o_s << std::setw(2) << std::setfill('0')
<< static_cast(*it);
}
return o_s.str();
}
else
{
return "";
}
}
bool Sha256::preprocessing(std::vector* _message) const
{
if (_message)
{
const uint64_t original_bit_size = _message->size() * 8;
//! 附加填充比特
size_t remainder = _message->size() % 64;
if (remainder < 56)
{
_message->push_back(0x80); // ox80 == 10000000
for (size_t i = 1; i < 56 - remainder; ++i)
{
_message->push_back(0x00);
}
}
else if (remainder == 56)
{
_message->push_back(0x80);
for (size_t i = 1; i < 64; ++i)
{
_message->push_back(0x00);
}
}
else
{
_message->push_back(0x80);
for (size_t i = 1; i < 64 - remainder + 56; ++i)
{
_message->push_back(0x00);
}
}
//! 附加原始文本的长度值
for (int i = 1; i <= 8; ++i)
{
uint8_t c = static_cast(original_bit_size >> (64 - 8 * i));
_message->push_back(c);
}
return true;
}
else
{
return false;
}
}
bool Sha256::breakTextInto64ByteChunks(const std::vector& message,
std::vector>* _chunks) const
{
if (_chunks && 0 == message.size() % 64)
{
_chunks->clear(); // 清空输出buffer
size_t quotient = message.size() / 64;
for (size_t i = 0; i < quotient; ++i)
{
std::vector temp(message.begin() + i * 64, message.begin() + (i + 1) * 64);
_chunks->push_back(temp);
}
return true;
}
else
{
return false;
}
}
bool Sha256::structureWords(const std::vector& chunk,
std::vector* _words) const
{
if (_words && 64 == chunk.size())
{
_words->resize(64);
auto& words = *_words;
for (int i = 0; i < 16; ++i)
{
words[i] = (static_cast(chunk[i * 4]) << 24)
| (static_cast(chunk[i * 4 + 1]) << 16)
| (static_cast(chunk[i * 4 + 2]) << 8)
| static_cast(chunk[i * 4 + 3]);
}
for (int i = 16; i < 64; ++i)
{
words[i] = small_sigma1(words[i - 2])
+ words[i - 7]
+ small_sigma0(words[i - 15])
+ words[i - 16];
}
return true;
}
else
{
return false;
}
}
bool Sha256::transform(const std::vector& words,
std::vector* _message_digest) const
{
if (_message_digest && 8 == _message_digest->size() && 64 == words.size())
{
std::vector d = *_message_digest;
for (int i = 0; i < 64; ++i)
{
uint32_t temp1 = d[7] + big_sigma1(d[4]) + ch(d[4], d[5], d[6]) + add_constant_[i] + words[i];
uint32_t temp2 = big_sigma0(d[0]) + maj(d[0], d[1], d[2]);
d[7] = d[6];
d[6] = d[5];
d[5] = d[4];
d[4] = d[3] + temp1;
d[3] = d[2];
d[2] = d[1];
d[1] = d[0];
d[0] = temp1 + temp2;
}
for (int i = 0; i < 8; ++i)
{
(*_message_digest)[i] += d[i];
}
return true;
}
else
{
return false;
}
}
bool Sha256::produceFinalHashValue(const std::vector& input,
std::vector* _output) const
{
if (_output)
{
_output->clear();
for (auto it = input.begin(); it != input.end(); ++it)
{
for (int i = 0; i < 4; i++)
{
_output->push_back(static_cast((*it) >> (24 - 8 * i)));
}
}
return true;
}
else
{
return false;
}
}
} // namespace ly
3)main.cpp
/*
* Filename: main.cpp
* Author: L.Y.
* Brief: SHA256算法测试
*/
#include
#include
#include "sha256.hpp"
int main()
{
std::string message = "seek truth from facts"; // 待加密的信息
ly::Sha256 sha256; // 初始化Sha256对象
std::string message_digest = sha256.getHexMessageDigest(message); // 加密
std::cout << message_digest << std::endl; // 输出加密结果
getchar();
}