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 2009-08-13 17:32  小橋流水  阅读(436)  评论(0编辑  收藏  举报

导航