我是Delphi的新手。我正在尝试对ASCII字符串进行单向散列。有没有办法加密ASCII字符串而不将Indy或任何其他加密库添加到项目中并且只是通过标准函数?
如果有任何方法可以使用Windows API,我该怎么做?
如果有可能,请举个例子。
答案 0 :(得分:10)
您可以使用Windows加密库来实现此目的。例如,这是计算MD5哈希值的最小单位:
unit MD5;
interface
uses
SysUtils, Math, Windows, Classes;
type
TMD5Hash = array [0..15] of Byte;
function MD5Hash(Stream: TStream): TMD5Hash; overload;
function MD5Hash(const Bytes: TBytes): TMD5Hash; overload;
function MD5Hash(const Str: RawByteString): TMD5Hash; overload;
function MD5ToString(const Hash: TMD5Hash): string;
implementation
type
HCRYPTPROV = type NativeUInt;
HCRYPTKEY = type NativeUInt;
HCRYPTHASH = type NativeUInt;
ALG_ID = Cardinal;
const
PROV_RSA_FULL = 1;
CRYPT_VERIFYCONTEXT = $F0000000;
ALG_CLASS_HASH = 4 shl 13;
ALG_TYPE_ANY = 0;
ALG_SID_MD5 = 3;
CALG_MD5 = ALG_CLASS_HASH or ALG_TYPE_ANY or ALG_SID_MD5;
HP_HASHVAL = $0002;
function CryptAcquireContextW(
out phProv: HCRYPTPROV;
pszContainer: PWideChar;
pszProvider: PWideChar;
dwProvType: DWORD;
dwFlags: DWORD
): BOOL; stdcall; external 'Advapi32.dll';
function CryptReleaseContext(
hProv: HCRYPTPROV;
dwFlags: DWORD
): BOOL; stdcall; external 'Advapi32.dll';
function CryptCreateHash(
hProv: HCRYPTPROV;
Algid: ALG_ID;
hKey: HCRYPTKEY;
dwFlags: DWORD;
out phHash: HCRYPTHASH
): BOOL; stdcall; external 'Advapi32.dll';
function CryptDestroyHash(
hHash: HCRYPTHASH
): BOOL; stdcall; external 'Advapi32.dll';
function CryptHashData(
hHash: HCRYPTHASH;
pbData: Pointer;
dwDataLen: DWORD;
dwFlags: DWORD
): BOOL; stdcall; external 'Advapi32.dll';
function CryptGetHashParam(
hHash: HCRYPTHASH;
dwParam: DWORD;
pbData: Pointer;
var pdwDataLen: DWORD;
dwFlags: DWORD
): BOOL; stdcall; external 'Advapi32.dll';
function MD5Hash(Stream: TStream): TMD5Hash;
const
BuffSize = 1024;
var
hProv: HCRYPTPROV;
hHash: HCRYPTHASH;
Buff: array [0..BuffSize] of Byte;
BytesLeft: Int64;
BytesToRead: Integer;
HashSize: DWORD;
begin
Win32Check(CryptAcquireContextW(hProv, nil, nil, PROV_RSA_FULL, CRYPT_VERIFYCONTEXT));
try
Win32Check(CryptCreateHash(hProv, CALG_MD5, 0, 0, hHash));
try
BytesLeft := Stream.Size-Stream.Position;
while BytesLeft>0 do begin
BytesToRead := Min(BytesLeft, BuffSize);
Stream.ReadBuffer(Buff, BytesToRead);
Win32Check(CryptHashData(hHash, @Buff, BytesToRead, 0));
dec(BytesLeft, BytesToRead);
end;
HashSize := SizeOf(Result);
Win32Check(CryptGetHashParam(hHash, HP_HASHVAL, @Result, HashSize, 0));
finally
Win32Check(CryptDestroyHash(hHash));
end;
finally
Win32Check(CryptReleaseContext(hProv, 0));
end;
end;
function MD5Hash(const Str: RawByteString): TMD5Hash;
var
Stream: TStringStream;
begin
Stream := TStringStream.Create(Str);
try
Result := MD5Hash(Stream);
finally
Stream.Free;
end;
end;
function MD5Hash(const Bytes: TBytes): TMD5Hash;
var
Stream: TBytesStream;
begin
Stream := TBytesStream.Create(Bytes);
try
Result := MD5Hash(Stream);
finally
Stream.Free;
end;
end;
function MD5ToString(const Hash: TMD5Hash): string;
begin
SetLength(Result, SizeOf(Hash)*2);
BinToHex(Pointer(@Hash)^, PChar(Result), SizeOf(Hash));
end;
end.
答案 1 :(得分:2)
Zeid,我不认为它在加密方面是安全的,但是Delphi确实在System.Generics.Defaults单元中有一个名为BobJenkinsHash(!!!)的哈希函数。它也不是用户友好的(要求你提供你传递给它的内存块的长度(大小?)并返回一个整数值。但是,如果你的要求是1)Delphi和2)没有第三方库和3)你不想自己实现另一个哈希,我相信这是你唯一的选择。
function BobJenkinsHash(const Data; Len, InitData: Integer): Integer;
您需要将字符串作为第一个参数传递,字符串的长度* char的大小作为第二个参数,以及任何其他数字(通常为零)作为最后一个参数。