哈希算法并不是一种加密算法。不能对用户信息的保护。因为哈希算法是单向的,可以将任何大小的数据转化为定长的“指纹”,而且无法被反向计算。
但Hash算法常用于对口令的保存上。另外,即使数据源只改动了一丁点,哈希的结果也会完全不同。这样的特性使得它非常适合用于保存密码,因为我们需要加密后的密码无法被解密,同时也能保证正确校验每个用户的密码。
例如,用户登录网站需要通过用户名和密码来进行验证。如果网站后台直接保存用户的口令明文,一旦数据库发生泄漏,后果不堪设想。
然而,由于有时候用户设置的口令的强度不够,只是一些常见的字符串,如password、123456等。有人专门搜集这些常见的口令,计算对应的Hash值,制成字典。这样通过Hash值可以快速的查到原始口令。这一类型以空间换时间的攻击方法包括字典攻击和彩虹攻击(只保存一条Hash链的首尾值,相对字典攻击可以节省存储空间)等。
为了防范这一类攻击,一般采用加盐值(salt)的方法。这样保存的不是口令明文的Hash值,而是口令再加上一段随机字符串(即“盐”)之后的Hash值。盐是一个添加到用户的密码哈希过程中的一段随机序列。这个机制能够防止通过预先计算结果的彩虹表破解。
每个用户都有自己的盐,这样的结果就是即使用户的密码相同,通过加盐后哈希值也将不同。
为了校验密码是否正确,我们需要储存盐值。通常和密码哈希值一起存放在账户数据库中,或者直接存为哈希字符串的一部分。
import javax.crypto.SecretKeyFactory;
import javax.crypto.spec.PBEKeySpec;
import java.math.BigI