我有这个生成 HMAC 的PHP代码(而不是简单的消息摘要):
<?php
$key = "0123456789ABCDEF0123456789ABCDEF0123456789ABCDEF0123456789ABCDEF0123456789ABCDEF0123456789ABCDEF0123456789ABCDEF0123456789ABCDEF";
$binkey = pack("H*", $key);
echo strtoupper(hash_hmac('sha512', "ABC", $binkey));
?>
输入ABC
后,其输出为:
100A6A016A4B21AE120851D51C93B293D95B7D8A44B16ACBEFC2D1C9DF02B6F54FA3C2D6802E52FED5DF8652DDD244788A204682D2D1CE861FDA4E67F2792643
我需要在java中克隆它。
所以这是我目前的java克隆:
private String generateHMAC( String datas )
{
// final Charset asciiCs = Charset.forName( "utf-8" );
Mac mac;
String result = "";
try
{
byte[] bytesKey = PayboxConstants.KEY.getBytes( );
final SecretKeySpec secretKey = new SecretKeySpec( bytesKey, "HmacSHA512" );
mac = Mac.getInstance( "HmacSHA512" );
mac.init( secretKey );
final byte[] macData = mac.doFinal( datas.getBytes( ) );
byte[] hex = new Hex( ).encode( macData );
result = new String( hex, "ISO-8859-1" );
}
catch ( final NoSuchAlgorithmException e )
{
AppLogService.error( e );
}
catch ( final InvalidKeyException e )
{
AppLogService.error( e );
}
catch ( UnsupportedEncodingException e )
{
AppLogService.error( e );
}
return result.toUpperCase( );
}
但它没有成功,因为对于相同的输入(ABC),它的输出是:
AA6492987D7A7AC81109E877315414806F1973CC47B897ECE713171A25A11B279329B1BFF39EA72A5EFB7EDCD71D1F34D5AAC49999A780BD13F019ED99685B80
我尝试了很多其他java代码,但它们都不是php版本的精确克隆。
我做错了什么?
答案 0 :(得分:7)
您只是忘记在Java代码中模仿pack()
的行为(无论您需要什么)。
使用
final SecretKeySpec secretKey = new SecretKeySpec( DatatypeConverter.parseHexBinary(PayboxConstants.KEY), "HmacSHA512" );
在您的Java代码中。
DatatypeConverter.parseHexBinary()
来自JAXB API。
或者,如果您不想仅仅为了将HEX字符串转换为字节而包含JAXB,则可能需要使用已发布的代码here。
答案 1 :(得分:-1)
我在Java中将它用于SHA 512。它可能有所帮助:
public static String sha512 ( String str )
{
try
{
return sha512 ( str.getBytes ( "UTF-8" ) );
}
catch ( UnsupportedEncodingException e )
{
e.printStackTrace ( );
return "";
}
}
public static String sha512 ( byte[] array )
{
try
{
MessageDigest m = MessageDigest.getInstance ( "SHA-512" );
m.update ( array );
String hash = new BigInteger ( 1, m.digest ( ) ).toString ( 16 );
while ( hash.length ( ) < 32 )
{
hash = "0" + hash;
}
return hash;
}
catch ( NoSuchAlgorithmException e )
{
e.printStackTrace ( );
return "";
}
}
我还记得在stackoverflow中的帖子中有关于MD-5的详细答案(只是algorythm不同)
答案 2 :(得分:-1)
试试:
private String generateHMAC( String datas )
{
MessageDigest md = MessageDigest.getInstance("SHA-512");
md.update(datas.getBytes("UTF-8")); // Change this to "UTF-16" if needed
byte[] digest = md.digest();
StringBuffer hexString = new StringBuffer();
for (int i=0;i<digest.length;i++) {
hexString.append(Integer.toHexString(0xFF & digest[i]));
}
return hexString.toString();
}