如何在php内加密解密(PHP中散列密码的安全性分析)
如何在php内加密解密
PHP中散列密码的安全性分析本文实例讲述了PHP中散列密码的安全性。分享给大家供大家参考,具体如下:
php的基本哈希函数已经不再安全?
php手册中有专门的一个部分来介绍这个问题
http://php.net/manual/zh/faq.passwords.php
很多应用,都是将用户的密码都是直接通过md5加密直接存储到数据库中的,包括我最近在用的开源项目zabbix的web管理界面。
|
$password = "1234" ; $hash = md5( $password ); echo $res ; |
php常用的哈希函数有md5和sha1,这种哈希之后,一般是不可逆的,但是可以重现,也就是说同样的明文,哈希之后的结果是一样的,对于一些简单的明文,是可以通过遍历,然后对照加密之后的密文得到明文的。
网上有流传的“彩虹表”,就是遍历的到的一个非常大的数据库,存储了明文和密文的对照关系,通过查询就能得到密文对应的明文。
http://www.cmd5.com/
这个网站就提供这种服务器,也就说如果黑客“脱裤”成功,拿到用户密码的密文之后,还是有很大的可能性解密得到明文了。
将明文“1234”,通过md5加密之后,在上面的网站是很容易解密出来的。
通过“加盐”,增加破解难度
“加盐”的意思是给明文加上一些数据,然后再进行加密。这样的话,就算明文(用户的密码)比较简单,加盐之后就变得更加复杂一些,然后再加密,这就增加了黑客去解密明文的难度。
|
$password = "1234" ; $salt = "s@jn#.sK_jF3;gg*&" ; $hash = md5( $password . $salt ); echo $res ; |
同样的明文“1234”,加了一个比较复杂的“盐”之后,再进行加密,解密的难度就增加了不少,在上面的解密网站是就不能被解密出来了(最起码不付钱是解密不出来的,哈哈哈)。
上面我们对所有的密码都使用的同样的盐,这中方式是不大安全的。比如,张三和李四的密码是一样的,则存储在数据库中的密文也是一样的,这无疑让黑客更容易破解了。
更常使用的方式,是对于不同的用户使用不同的盐进行加密,在用户的注册过程中,生成用户对应的盐,然后进行存储;在用户登录时,取出盐用于加密操作,盐和用户id一一对应。
可以使用php自带的random_bytes
生成一定长度的盐
|
$password = "1234" ; $salt = bin2hex(random_bytes(32)); $hash = md5( $password . $salt ); echo $res ; |
关于盐的存储
可以将盐和密文一起存在数据库的用户信息表中,优点是数据库查询取出密码的同时也可以取出盐,进行加密比对操作,一次数据查询就可以搞定,缺点是安全性差,如果黑客“脱裤”成功,则获取密文的同时也获取了对应的盐。
更好的方案是将盐和密文分开存储,比如密文存储在mysql数据库中,盐存储在redis服务器中,这样即使黑客“脱裤”拿到了数据库中的密文,也需要再进一步拿到对应的盐才能进一步破解,安全性更好,不过这样需要进行二次查询,即每次登陆都需要从redis中取出对应的盐,牺牲了一定的性能,提高了安全性。
php5.5中更加安全的解决方案
说php是专为为web设计的语言一点也没错,应该是php开发者也注意到了这个密码保存的问题。
于是php5.5开始,就设计了password_hashing模块,用于密码的哈希和验证。
http://php.net/manual/zh/book.password.php
使用password_hash进行哈希,使用的算法、cost 和盐值作为哈希的一部分返回,所以不用单独保存salt的值,因为它每次都会自己生成salt,所以优点就是“每次加密的结果都不一样”,但是可以放心,加密结果包含了salt信息,password_verify可以正确解析。
|
$password = "1234" ; $hash = password_hash( $password ,PASSWORD_DEFAULT); |
哈希之后的结果,只能使用password_verify
进行验证,因此验证密码的功能只能由php语言来实现。
|
$password = "1234" ; $hash = password_hash( $password ,PASSWORD_DEFAULT); $res = password_verify( $password , $hash ); //验证结果为true |
优缺点分析
优点是安全性很高,即使被脱裤,也很难将密文解密,因为同一个密文,每次加密的结果都不一样,所以没法撞库!
password_hash实际上是对crypt和salt的封装,crypt加密比普通的md5和sha1更加复杂,所以耗时也更加多一些,这可以算是一个缺点,对于用户量很大,经常需要进行登录操作的站点,可能会有性能上的影响。还有一点是通用性不强,因为这种方式只适用于php语言,其他语言是没有办法对密文进行操作的。
刚才测试了一下password_hash的性能,吓的半死。。
md5.php
|
<?php $stime = microtime(true); $password = "root123@" ; $salt = "83979fklsdfgklu9023*&*(&()#&*(Y*(@&*<:L:%:::>><??11!!^%^$%$%^<>YUIYUIhjkdshfJKH#J#HJK#HKl;dskfs" ; for ( $i =0; $i <100; $i ++){ $res = md5( $password ); } $etime = microtime(true); echo "stime:$stime<br/>" ; echo "etime:$etime<br/>" ; echo "cost:" .( $etime - $stime ); |
运行结果:
stime:1478265603.1118
etime:1478265603.1229
cost:0.011116981506348
password_hash.php
|
<?php $stime = microtime(true); $password = "root123@" ; for ( $i =0; $i <100; $i ++){ $res = password_hash( $password ,PASSWORD_DEFAULT); } $etime = microtime(true); echo "stime:$stime<br/>" ; echo "etime:$etime<br/>" ; echo "cost:" .( $etime - $stime ); |
运行结果:
stime:1478265640.382
etime:1478265646.6675
cost:6.2854981422424
如果是安全性要求特别高的情况下,可以使用password_hash
的方式,这种情况下一般可以通过其他方式提高服务器性能。
不过,大多是情况下,将salt存储在redis,md5之后的密文存储在mysql的方式已经非常安全了,微笑 :)
希望本文所述对大家PHP程序设计有所帮助。
原文链接:https://blog.csdn.net/koastal/article/details/53039130
- phpredis怎么测试成功(PHP+redis实现的限制抢购防止商品超发功能详解)
- phpinclude的使用方法(PHP defined函数的使用图文详解)
- php常见的数组函数(用php定义一个数组最简单的方法)
- php解压压缩包(PHP 实现文件压缩解压操作的方法)
- phplaravel快速开发平台(laravel框架实现去掉URL中index.php的方法)
- nginx 怎么搭建web服务器(Linux+Nginx+Php架设高性能WEB服务器)
- php生成随机数讲解(PHP生成随机字符串实例代码字母+数字)
- 解释php中可变变量并举例说明(PHP Primary script unknown 解决方法总结)
- php生成安全随机数(PHP随机数函数rand与mt_rand的讲解)
- thinkphp返回json(thinkphp5框架实现数据库读取的数据转换成json格式示例)
- thinkphp框架切换数据库(ThinkPHP3.2.3框架实现执行原生SQL语句的方法示例)
- php守护进程(什么是PHP7中的孤儿进程与僵尸进程)
- php换行乱码(php输出文字乱码的解决方法)
- 常见的php五大运行模式详解(php设计模式之职责链模式定义与用法经典示例)
- php建造者模式实例(PHP工厂模式、单例模式与注册树模式实例详解)
- php面向对象怎么用(PHP面向对象程序设计之对象克隆clone和魔术方法__clone用法分析)
- 这个全椒人被通报表彰,看看你认识吗(这个全椒人被通报表彰)
- 全椒人,38年集体回忆 1980-2018 ,看完不要哭(全椒人38年集体回忆)
- 董元奔吟咏历代文人 1012新旧均可 全椒人张璪 1022 -1093(董元奔吟咏历代文人)
- 泪目 这位 刷屏 的英雄,是全椒人的骄傲(泪目这位刷屏)
- 人从众 火炎焱 全椒再现 正月十六走太平 的魅力(人从众火炎焱全椒再现)
- 官宣 全椒籍明星许海峰 奚秀兰 方芳 王璐瑶携手回家 走太平(全椒籍明星许海峰)
热门推荐
- sqlserver纵表变横表(SQL Server行转列的方法解析)
- pythonpandas提取列索引(Python3.5 Pandas模块缺失值处理和层次索引实例详解)
- laravel队列是如何触发的(laravel 事件/监听器实例代码)
- numpy如何获取索引(NumPy 基本切片和索引的具体使用方法)
- 如何用python处理excel表格(零基础使用Python读写处理Excel表格的方法)
- mysql随机获取数据
- dedecms修改教程(还原dedecms数据后系统基本参数空白的解决方法)
- nginx反向代理及原理(传说中的反向代理,Nginx+Apache软件配置Web服务器)
- thinkphp实战教程之博客技术学习(python3编写ThinkPHP命令执行Getshell的方法)
- mysql中常用的三种存储引擎的区别(MySQL 存储过程的优缺点分析)
排行榜
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9