java中散列是,如何在Java中散列密码?

实际上,您可以使用Java运行时内置的工具来执行此操作 . Java 6中的SunJCE支持PBKDF2,这是一种用于密码散列的好算法 .

byte[] salt = new byte[16];

random.nextBytes(salt);

KeySpec spec = new PBEKeySpec("password".toCharArray(), salt, 65536, 128);

SecretKeyFactory f = SecretKeyFactory.getInstance("PBKDF2WithHmacSHA1");

byte[] hash = f.generateSecret(spec).getEncoded();

Base64.Encoder enc = Base64.getEncoder();

System.out.printf("salt: %s%n", enc.encodeToString(salt));

System.out.printf("hash: %s%n", enc.encodeToString(hash));

这是一个可用于PBKDF2密码验证的实用程序类:

import java.security.NoSuchAlgorithmException;

import java.security.SecureRandom;

import java.security.spec.InvalidKeySpecException;

import java.security.spec.KeySpec;

import java.util.Arrays;

import java.util.Base64;

import java.util.regex.Matcher;

import java.util.regex.Pattern;

import javax.crypto.SecretKeyFactory;

import javax.crypto.spec.PBEKeySpec;

/**

* Hash passwords for storage, and test passwords against password tokens.

*

* Instances of this class can be used concurrently by multiple threads.

*

* @author erickson

* @see StackOverflow

*/

public final class PasswordAuthentication

{

/**

* Each token produced by this class uses this identifier as a prefix.

*/

public static final String ID = "$31$";

/**

* The minimum recommended cost, used by default

*/

public static final int DEFAULT_COST = 16;

private static final String ALGORITHM = "PBKDF2WithHmacSHA1";

private static final int SIZE = 128;

private static final Pattern layout = Pattern.compile("\\$31\\$(\\d\\d?)\\$(.{43})");

private final SecureRandom random;

private final int cost;

public PasswordAuthentication()

{

this(DEFAULT_COST);

}

/**

* Create a password manager with a specified cost

*

* @param cost the exponential computational cost of hashing a password, 0 to 30

*/

public PasswordAuthentication(int cost)

{

iterations(cost); /* Validate cost */

this.cost = cost;

this.random = new SecureRandom();

}

private static int iterations(int cost)

{

if ((cost < 0) || (cost > 30))

throw new IllegalArgumentException("cost: " + cost);

return 1 << cost;

}

/**

* Hash a password for storage.

*

* @return a secure authentication token to be stored for later authentication

*/

public String hash(char[] password)

{

byte[] salt = new byte[SIZE / 8];

random.nextBytes(salt);

byte[] dk = pbkdf2(password, salt, 1 << cost);

byte[] hash = new byte[salt.length + dk.length];

System.arraycopy(salt, 0, hash, 0, salt.length);

System.arraycopy(dk, 0, hash, salt.length, dk.length);

Base64.Encoder enc = Base64.getUrlEncoder().withoutPadding();

return ID + cost + '$' + enc.encodeToString(hash);

}

/**

* Authenticate with a password and a stored password token.

*

* @return true if the password and token match

*/

public boolean authenticate(char[] password, String token)

{

Matcher m = layout.matcher(token);

if (!m.matches())

throw new IllegalArgumentException("Invalid token format");

int iterations = iterations(Integer.parseInt(m.group(1)));

byte[] hash = Base64.getUrlDecoder().decode(m.group(2));

byte[] salt = Arrays.copyOfRange(hash, 0, SIZE / 8);

byte[] check = pbkdf2(password, salt, iterations);

int zero = 0;

for (int idx = 0; idx < check.length; ++idx)

zero |= hash[salt.length + idx] ^ check[idx];

return zero == 0;

}

private static byte[] pbkdf2(char[] password, byte[] salt, int iterations)

{

KeySpec spec = new PBEKeySpec(password, salt, iterations, SIZE);

try {

SecretKeyFactory f = SecretKeyFactory.getInstance(ALGORITHM);

return f.generateSecret(spec).getEncoded();

}

catch (NoSuchAlgorithmException ex) {

throw new IllegalStateException("Missing algorithm: " + ALGORITHM, ex);

}

catch (InvalidKeySpecException ex) {

throw new IllegalStateException("Invalid SecretKeyFactory", ex);

}

}

/**

* Hash a password in an immutable {@code String}.

*

*

Passwords should be stored in a {@code char[]} so that it can be filled

* with zeros after use instead of lingering on the heap and elsewhere.

*

* @deprecated Use {@link #hash(char[])} instead

*/

@Deprecated

public String hash(String password)

{

return hash(password.toCharArray());

}

/**

* Authenticate with a password in an immutable {@code String} and a stored

* password token.

*

* @deprecated Use {@link #authenticate(char[],String)} instead.

* @see #hash(String)

*/

@Deprecated

public boolean authenticate(String password, String token)

{

return authenticate(password.toCharArray(), token);

}

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值