我如何在Java中哈希一些String与SHA-256 ?
String
SHA-256
SHA-256不是一种“编码”。-这是一个单向哈希。
你基本上将字符串转换为字节(例如使用text.getBytes(StandardCharsets.UTF_8)),然后哈希字节。注意,哈希的结果也是任意二进制数据,如果你想在字符串中表示它,你应该使用base64或hex…不尝试使用String(byte[], String)构造函数。
text.getBytes(StandardCharsets.UTF_8)
String(byte[], String)
如。
MessageDigest digest = MessageDigest.getInstance("SHA-256"); byte[] hash = digest.digest(text.getBytes(StandardCharsets.UTF_8));
完整的例子哈希字符串作为另一个字符串。
public static String sha256(final String base) { try{ final MessageDigest digest = MessageDigest.getInstance("SHA-256"); final byte[] hash = digest.digest(base.getBytes("UTF-8")); final StringBuilder hexString = new StringBuilder(); for (int i = 0; i < hash.length; i++) { final String hex = Integer.toHexString(0xff & hash[i]); if(hex.length() == 1) hexString.append('0'); hexString.append(hex); } return hexString.toString(); } catch(Exception ex){ throw new RuntimeException(ex); } }
我认为最简单的解决方案是使用Apache Common Codec:
String sha256hex = org.apache.commons.codec.digest.DigestUtils.sha256Hex(stringText);
另一个替代方法是番石榴,它有一个易于使用的哈希实用程序套件。例如,使用SHA256作为十六进制字符串来哈希字符串,你可以简单地这样做:
final String hashed = Hashing.sha256() .hashString("your input", StandardCharsets.UTF_8) .toString();
import java.security.MessageDigest; public class CodeSnippets { public static String getSha256(String value) { try{ MessageDigest md = MessageDigest.getInstance("SHA-256"); md.update(value.getBytes()); return bytesToHex(md.digest()); } catch(Exception ex){ throw new RuntimeException(ex); } } private static String bytesToHex(byte[] bytes) { StringBuffer result = new StringBuffer(); for (byte b : bytes) result.append(Integer.toString((b & 0xff) + 0x100, 16).substring(1)); return result.toString(); } }
我通过DigestUtils跟踪Apache代码,sha256似乎默认返回java.security.MessageDigest进行计算。Apache没有实现独立的sha256解决方案。我正在寻找一个独立的实现来与java.security库进行比较。仅供参考。
DigestUtils
sha256
java.security.MessageDigest
java.security
如果你使用的是Java 8,你可以对byte[]进行编码
byte[]
MessageDigest digest = MessageDigest.getInstance("SHA-256"); byte[] hash = digest.digest(text.getBytes(StandardCharsets.UTF_8)); String encoded = Base64.getEncoder().encodeToString(hash);
String hashWith256(String textToHash) { MessageDigest digest = MessageDigest.getInstance("SHA-256"); byte[] byteOfTextToHash = textToHash.getBytes(StandardCharsets.UTF_8); byte[] hashedByetArray = digest.digest(byteOfTextToHash); String encoded = Base64.getEncoder().encodeToString(hashedByetArray); return encoded; }
您可以通过以下方式使用MessageDigest:
public static String getSHA256(String data){ StringBuffer sb = new StringBuffer(); try{ MessageDigest md = MessageDigest.getInstance("SHA-256"); md.update(data.getBytes()); byte byteData[] = md.digest(); for (int i = 0; i < byteData.length; i++) { sb.append(Integer.toString((byteData[i] & 0xff) + 0x100, 16).substring(1)); } } catch(Exception e){ e.printStackTrace(); } return sb.toString(); }
下面是一种将摘要转换为十六进制字符串的更有效的方法:
private static final char[] hexArray = "0123456789abcdef".toCharArray(); public static String getSHA256(String data) { StringBuilder sb = new StringBuilder(); try { MessageDigest md = MessageDigest.getInstance("SHA-256"); md.update(data.getBytes()); byte[] byteData = md.digest(); sb.append(bytesToHex(byteData); } catch(Exception e) { e.printStackTrace(); } return sb.toString(); } private static String bytesToHex(byte[] bytes) { char[] hexChars = new char[bytes.length * 2]; for ( int j = 0; j < bytes.length; j++ ) { int v = bytes[j] & 0xFF; hexChars[j * 2] = hexArray[v >>> 4]; hexChars[j * 2 + 1] = hexArray[v & 0x0F]; } return String.valueOf(hexChars); }
有人知道Java中更快的方法吗?
这是我使用Kotlin的方法:
private fun getHashFromEmailString(email : String) : String{ val charset = Charsets.UTF_8 val byteArray = email.toByteArray(charset) val digest = MessageDigest.getInstance("SHA-256") val hash = digest.digest(byteArray) return hash.fold("", { str, it -> str + "%02x".format(it)}) }
private static String getMessageDigest(String message, String algorithm) { MessageDigest digest; try { digest = MessageDigest.getInstance(algorithm); byte data[] = digest.digest(message.getBytes("UTF-8")); return convertByteArrayToHexString(data); } catch (NoSuchAlgorithmException | UnsupportedEncodingException e) { // TODO Auto-generated catch block e.printStackTrace(); } return null; }
你可以像下面这样用不同的算法调用上面的方法。
getMessageDigest(message, "MD5"); getMessageDigest(message, "SHA-256"); getMessageDigest(message, "SHA-1");
你可以引用这个链接来获取完整的应用程序。
在Java中,MessageDigest类用于计算加密哈希值。这个类提供了加密哈希函数(MD5, sha - 1和sha - 256)来查找文本的哈希值。
使用SHA-256算法的代码示例。
public void printHash(String str) throws NoSuchAlgorithmException { MessageDigest md=MessageDigest.getInstance("SHA-256"); byte[] sha256=md.digest(str.getBytes(StandardCharsets.UTF_8)); for(byte b : sha256){ System.out.printf("%02x",b); } }
在Java 8中
import java.nio.charset.StandardCharsets; import java.security.MessageDigest; import java.security.NoSuchAlgorithmException; import java.util.Scanner; import javax.xml.bind.DatatypeConverter; Scanner scanner = new Scanner(System.in); String password = scanner.nextLine(); scanner.close(); MessageDigest digest = null; try { digest = MessageDigest.getInstance("SHA-256"); } catch (NoSuchAlgorithmException e) { // TODO Auto-generated catch block e.printStackTrace(); } byte[] hash = digest.digest(password.getBytes(StandardCharsets.UTF_8)); String encoded = DatatypeConverter.printHexBinary(hash); System.out.println(encoded.toLowerCase());
这是我用于哈希的:
String pass = "password"; MessageDigest messageDigest = MessageDigest.getInstance("SHA-256"); byte hashBytes[] = messageDigest.digest(pass.getBytes(StandardCharsets.UTF_8)); BigInteger noHash = new BigInteger(1, hashBytes); String hashStr = noHash.toString(16);
输出:5 e884898da28047151d0e56f8dc6292773603d0d6aabbdd62a11ef721d1542d8
sha-256
hex
public static String sha256(final String data) { try { final byte[] hash = MessageDigest.getInstance("SHA-256").digest(data.getBytes(StandardCharsets.UTF_8)); final StringBuilder hashStr = new StringBuilder(hash.length); for (byte hashByte : hash) hashStr.append(Integer.toHexString(255 & hashByte)); return hashStr.toString(); } catch (NoSuchAlgorithmException e) { e.printStackTrace(); return null; } }
这个方法返回一个带有0的左填充String:
Java 10和之后:
Java 10
public static String sha256(String text) { try { var messageDigest = MessageDigest.getInstance("SHA-256"); var hash = messageDigest.digest(text.getBytes(StandardCharsets.UTF_8)); return String.format("%064x", new BigInteger(1, hash)); } catch (NoSuchAlgorithmException e) { e.printStackTrace(); return null; } }
Java 8:
Java 8
public static String sha256(String text) { try { MessageDigest messageDigest = MessageDigest.getInstance("SHA-256"); byte[] hash = messageDigest.digest(text.getBytes(StandardCharsets.UTF_8)); return String.format("%064x", new BigInteger(1, hash)); } catch (NoSuchAlgorithmException e) { e.printStackTrace(); return null; } }
顺便说一句,你可以使用"%064X"作为大写的结果。
"%064X"
例子:
System.out.println(sha256("hello world 1"));
063年dbf1d36387944a5f0ace625b4d3ee36b2daefd8bdaee5ede723637efb1cf4
与Linux cmd的比较:
$ echo -n 'hello world 1' | sha256sum 063 dbf1d36387944a5f0ace625b4d3ee36b2daefd8bdaee5ede723637efb1cf4 - < / p >