SHA256算法C++实现

一、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();
}

你可能感兴趣的:(C++,SHA256,C++,SHA256,实现)