详解PHP处理密码的几种方式


Posted in PHP onNovember 30, 2016

在 PHP中,经常会对用户身份进行认证。本文意在讨论对密码的处理,也就是对密码的加密处理。

MD5

相信很多PHP开发者在最先接触PHP的时候,处理密码的首选加密函数可能就是MD5了,我当时就是这样的:

$password = md5($_POST["password"]);

上面这段代码是不是很熟悉?然而MD5的加密方式目前已经不太安全了,因为它的加密算法实在是显得有点简单了,而且很多破解密码的站点都存放了很多经过MD5加密的密码字符串,所以这里我是非常不提倡还在单单使用MD5来加密用户的密码的。

SHA256 和 SHA512

其实跟前面的MD5同期的还有一个SHA1加密方式的,不过也是算法比较简单,所以这里就不介绍了。而这里即将要说到的SHA256 和 SHA512都是来自于SHA2家族的加密函数,看名字可能你就猜的出来了,这两个加密方式分别生成256和512比特长度的 hash字串。

他们的使用方法如下:

$password = hash("sha256", $password);

PHP内置了hash()函数,你只需要将加密方式传给hash()函数就好了。你可以直接指明sha256, sha512, md5, sha1等加密方式。

盐值

在加密的过程,我们还有一个非常常见的东西:盐值。对,我们在加密的时候其实会给加密的字符串添加一个额外的字符串,以达到提高一定安全的目的,并且盐值要记录下来,方便以后的比对:

function generateHashWithSalt($password) {
  $intermediateSalt = md5(uniqid(rand(), true));
  $salt = substr($intermediateSalt, 0, 6); 
  return hash("sha256", $password . $salt);
}

Bcrypt

Bcrypt不失为一种比较不错的加密方式了,但是后面介绍的 Hashing API更好。

function generateHash($password) {
  if (defined("CRYPT_BLOWFISH") && CRYPT_BLOWFISH) {
    $salt = '$2y$11$' . substr(md5(uniqid(rand(), true)), 0, 22);
    return crypt($password, $salt);
  }
}

Bcrypt 其实就是Blowfish和crypt()函数的结合,我们这里通过CRYPT_BLOWFISH判断Blowfish是否可用,然后像上面一样生成一个盐值,不过这里需要注意的是,crypt()的盐值必须以 $2a$ 或者 $2y$ 开头。

Password Hashing API

这里才是我们的重头戏,Password Hashing API是PHP 5.5之后才有的新特性,它主要是提供下面几个函数供我们使用:

password_hash()     //对密码加密.
password_verify()    //验证已经加密的密码,检验其hash字串是否一致.
password_needs_rehash() //给密码重新加密.
password_get_info()   //返回加密算法的名称和一些相关信息.

使用这套 API不仅简单,而且更加安全,这也是 PHP官方推荐的加密方式。

$hash = password_hash($passwod, PASSWORD_DEFAULT);

PASSWORD_DEFAULT目前使用的就是Bcrypt加密算法,这里需要注意的是,如果你代码使用的都是PASSWORD_DEFAULT加密方式,那么在数据库的表中,password字段就得设置超过60个字符长度,你也可以使用PASSWORD_BCRYPT算法,该算法加密后的字符串长度总为60。

这里使用 password_hash()你完全可以不提供盐值(salt)和 消耗值 (cost),你可以将后者理解为一种性能的消耗值,cost越大,加密算法越复杂,消耗的内存也就越大。当然,如果你需要指定对应的盐值和消耗值,你可以这样写:

$options = [
  'salt' => custom_function_for_salt(), //自定义函数来获得盐值
  'cost' => 12 // the default cost is 10
];
$hash = password_hash($password, PASSWORD_DEFAULT, $options);

不过一般自定义 cost就好了, salt值则使用默认的。

加密好后,只需要简单的使用即可验证密码是否正确

<?php
if (password_verify($password, $hash)) {
  // Pass
}
else {
  // Invalid
}

直接使用password_verify就可以对我们之前加密过的字符串(存在数据库中)进行验证了。

如果要更换更改加密的方式,则必须使用以下代码来重新加密:

if (password_needs_rehash($hash, PASSWORD_DEFAULT, ['cost' => 12])) {
  // cost 变为 12
  $hash = password_hash($password, PASSWORD_DEFAULT, ['cost' => 12]);

  // 然后重新保存 hash值
}

只有这样,PHP的 Password Hashing API才会知道我们重现更换了加密方式,这样才能完成之后的密码验证。

password_get_info(),这个函数一般可以看到下面三个信息:

1、algo ? 算法实例

2、algoName ? 算法名字

3、options ? 加密时候的可选参数

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

PHP 相关文章推荐
新浪新闻小偷
Oct 09 PHP
php file_exists 检查文件或目录是否存在的函数
May 10 PHP
php如何调用webservice应用介绍
Nov 24 PHP
php解决抢购秒杀抽奖等大流量并发入库导致的库存负数的问题
Jun 19 PHP
ThinkPHP的URL重写问题
Jun 22 PHP
php将HTML表格每行每列转为数组实现采集表格数据的方法
Apr 03 PHP
php实现求相对时间函数
Jun 15 PHP
yii2利用自带UploadedFile实现上传图片的示例
Feb 16 PHP
基于Codeigniter框架实现的student信息系统站点动态发布功能详解
Mar 23 PHP
深入解析Laravel5.5中的包自动发现Package Auto Discovery
Sep 13 PHP
PHP环形链表实现方法示例
Sep 15 PHP
PHP 结合 Boostrap 结合 js 实现学生列表删除编辑及搜索功能
May 21 PHP
php+js实现百度地图多点标注的方法
Nov 30 #PHP
php 运算符与表达式详细介绍
Nov 30 #PHP
PHP AjaxForm提交图片上传并显示图片源码
Nov 29 #PHP
php判断是否为ajax请求的方法
Nov 29 #PHP
PHP判断文件是否被引入的方法get_included_files用法示例
Nov 29 #PHP
php获取开始与结束日期之间所有日期的方法
Nov 29 #PHP
PHP精确计算功能示例
Nov 29 #PHP
You might like
利用php+mysql来做一个功能强大的在线计算器
2010/10/12 PHP
PHP数据库连接mysql与mysqli对比分析
2016/01/04 PHP
php利用fsockopen GET/POST提交表单及上传文件
2017/05/22 PHP
yii插入数据库防并发的简单代码
2017/05/27 PHP
PHP 进程池与轮询调度算法实现多任务的示例代码
2019/11/26 PHP
基于jquery的兼容各种浏览器的iframe自适应高度的脚本
2010/08/13 Javascript
面向对象设计模式的核心法则
2013/11/10 Javascript
javascript实现博客园页面右下角返回顶部按钮
2015/02/22 Javascript
JS实现控制表格只显示行边框或者只显示列边框的方法
2015/03/31 Javascript
基于jQuery+Cookie实现的防止刷新的在线考试倒计时
2015/06/19 Javascript
jquery实现Slide Out Navigation滑出式菜单效果代码
2015/09/07 Javascript
正则表达式(语法篇推荐)
2016/06/24 Javascript
vue组件之间的数据传递方法详解
2019/04/19 Javascript
Vue中强制组件重新渲染的正确方法
2021/01/03 Vue.js
bat和python批量重命名文件的实现代码
2016/05/19 Python
Python读取图片属性信息的实现方法
2016/09/11 Python
Python基于xlrd模块操作Excel的方法示例
2018/06/21 Python
python网络应用开发知识点浅析
2019/05/28 Python
python3.4+pycharm 环境安装及使用方法
2019/06/13 Python
浅谈tensorflow中张量的提取值和赋值
2020/01/19 Python
树莓派升级python的具体步骤
2020/07/05 Python
使用phonegap克隆和删除联系人的实现方法
2017/03/31 HTML / CSS
享誉全球的多元化时尚精品购物平台:Farfetch发发奇(支持中文)
2017/08/08 全球购物
REN Clean Skincare官网:英国本土有机护肤品牌
2019/02/23 全球购物
如何现实servlet的单线程模式
2014/08/05 面试题
人力资源经理自我评价
2014/01/04 职场文书
最经典的大学生职业生涯规划范文
2014/03/05 职场文书
银行委托书范本
2014/04/04 职场文书
公务员诚信承诺书
2014/05/26 职场文书
数据结构课程设计心得体会
2016/01/15 职场文书
jQuery实现广告显示和隐藏动画
2021/07/04 jQuery
Mysql8.0递归查询的简单用法示例
2021/08/04 MySQL
详细聊聊关于Mysql联合查询的那些事儿
2021/10/24 MySQL
gojs实现蚂蚁线动画效果
2022/02/18 Javascript
动画《朋友游戏》公开佐藤友生绘制的开播纪念绘
2022/04/06 日漫
docker 制作mysql镜像并自动安装
2022/05/20 Servers