VC HASH函数

    在程序很多地方都需要使用到MD5与SHA1之类的HASH函数,当然你可以自己实现,或去网上找实现的源码,但如果你在VC环境中的话,你就没有必要了,直接可以调用CAPI了,在程序中加上的头文件#include <wincrypt.h>

    下面的函数可以完成计算Hash的功能:

// 计算Hash,成功返回0,失败返回GetLastError()
//  CONST BYTE *pbData    输入数据 
//  DWORD dwDataLen        输入数据字节长度 
//  ALG_ID algId        Hash 算法:CALG_MD5,CALG_SHA
//  BYTE* pHash            输出hash值
//
DWORD ComputeHash(CONST BYTE *pbData, DWORD dwDataLen, ALG_ID algId, BYTE*& pbOutHash, DWORD& dwHashLen)
{
    DWORD dwReturn = 0;
    HCRYPTPROV hProv;
    if (!CryptAcquireContext(&hProv, NULL, NULL, PROV_RSA_FULL, CRYPT_VERIFYCONTEXT))
        return (dwReturn = GetLastError());

    HCRYPTHASH hHash;

    //Alg Id:CALG_MD5,CALG_SHA
    if(!CryptCreateHash(hProv, algId, 0, 0, &hHash)) 
    {
        dwReturn = GetLastError();
        CryptReleaseContext(hProv, 0);
        return dwReturn;
    }

    if(!CryptHashData(hHash, pbData, dwDataLen, 0))
    {
        dwReturn = GetLastError();
        CryptDestroyHash(hHash);
        CryptReleaseContext(hProv, 0);
        return dwReturn;
    }

    DWORD dwLen = sizeof(dwHashLen);
    CryptGetHashParam(hHash, HP_HASHSIZE, (BYTE*)(&dwHashLen), &dwLen, 0);

    pbOutHash = new BYTE[dwHashLen];
    dwLen = dwHashLen;
    CryptGetHashParam(hHash, HP_HASHVAL, pbOutHash, &dwLen, 0);

    CryptDestroyHash(hHash);
    CryptReleaseContext(hProv, 0);
    return dwReturn;

下面是测试代码:

int _tmain(int argc, TCHAR* argv[], TCHAR* envp[])
{
    int nRetCode = 0;

    // 初始化 MFC 并在失败时显示错误
    if (!AfxWinInit(::GetModuleHandle(NULL), NULL, ::GetCommandLine(), 0))
    {
        // TODO: 更改错误代码以符合您的需要
        _tprintf(_T("错误: MFC 初始化失败\n"));
        nRetCode = 1;
    }
    else
    {
        // TODO: 在此处为应用程序的行为编写代码。
        wstring s= _T("你好,你在哪里ss");
        string result;
        UnicodeToOther(s, result, CodePages::utf_8);
        BYTE* hashData = (BYTE*)result.c_str();
        BYTE* hash;
        DWORD hashLen;
        ComputeHash(hashData, (int)result.length() - 1, CALG_SHA1, hash, hashLen); // 这里一定要-1
        for(int i = 0; i < hashLen; i ++)
        {
            cout<<(unsigned int)hash[i]<<",";
        }
    }

    return nRetCode;
经试验与.NET中的计算结果一致。

posted on   小橋流水  阅读(439)  评论(0编辑  收藏  举报

编辑推荐:
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
阅读排行:
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· Ollama——大语言模型本地部署的极速利器
· [AI/GPT/综述] AI Agent的设计模式综述

导航

统计

点击右上角即可分享
微信分享提示