## TP5实战开发前篇---缓存(Cache) PHP缓存技术的应用时相当普遍的,也许有些人还对这项技术不太了解,我们现在就为大家详细的介绍一下PHP缓存技术的相关应用技巧。   在大部份情况下我们的网站都会使用数据库作为站点数据存储的容器。当你执行一个SQL查询时,典型的处理过程是:连接数据库->准备SQL查询->发送查询到数据库->取得数据库返回结果->关闭数据库连接。但数据库中有些数据是完全静态的或不太经常变动的,缓存系统会通过把SQL查询的结果缓存到一个更快的存储系统中存储,从而避免频繁操作数据库而很大程度上提高了程序执行时间,而且缓存查询结果也允许你后期处理。 普遍使用的PHP缓存技术 ### php缓存技术之数据缓存:   这里所说的数据缓存是指数据库查询缓存,每次访问页面的时候,都会先检测相应的缓存数据是否存在,如果不存在,就连接数据库,得到数据,并把查询结果序列化后保存到文件中,以后同样的查询结果就直接从缓存文件中获得。 ### PHP缓存技术之页面缓存:   每次访问页面的时候,都会先检测相应的缓存页面文件是否存在,如果不存在,就连接数据库,得到数据,显示页面并同时生成缓存页面文件,这样下次访问的时候页面文件就发挥作用了。(模板引擎和网上常见的一些缓存类通常有此功能) ### PHP缓存技术之内存缓存:   在里就不介绍了,不是本文所要讨论的,只简单提一下:   Memcached是高性能的,分布式的内存对象缓存系统,用于在动态应用中减少数据库负载,提升访问速度。   dbcached 是一款基于 Memcached 和 NMDB 的分布式 key-value 数据库内存缓存系统。   以上的缓存技术虽然能很好的解决频繁查询数据库的问题,但其缺点在在于数据无时效性,下面我给出我在项目中常用的方法: ### PHP缓存技术之时间触发缓存:   检查文件是否存在并且时间戳小于设置的过期时间,如果文件修改的时间戳比当前时间戳减去过期时间戳大,那么就用缓存,否则更新缓存。   设定时间内不去判断数据是否要更新,过了设定时间再更新缓存。以上只适合对时效性要求不高的情况下使用,否则请看下面。 ### PHP缓存技术之内容触发缓存:   当插入数据或更新数据时,强制更新缓存。   在这里我们可以看到,当有大量数据频繁需要更新时,最后都要涉及磁盘读写操作。怎么解决呢?我在日常项目中,通常并不缓存所有内容,而是缓存一部分不经常变的内容来解决。但在大负荷的情况下,最好要用共享内存做缓存系统。   到这里PHP缓存也许有点解决方案了,但其缺点是,因为每次请求仍然要经过PHP解析,在大负荷的情况下效率问题还是比效严重,在这种情况下,也许会用到静态缓存。 ### PHP缓存技术之静态缓存   这里所说的静态缓存是指HTML缓存,HTML缓存一般是无需判断数据是否要更新的,因为通常在使用HTML的场合一般是不经常变动内容的页面。数据更新的时候把HTML也强制更新一下就可以了。 > 以上内容来源于网络: PHP缓存技术的使用技巧分享 # 缓存应用 源码实例 >[info] 缓存除了以上对以上减轻服务器压力提高程序性能效率外,还能实现分布式服务器群一些特殊功效。比如说我们使用Ng的反向代理建立了一个服务器群,那么我们如何做到多服务器能获取到相同客户端相同的身份信息呢,很显然session是不能解决这个问题的 。这时候有一台共享的缓存服务器就是最便捷的解决方案。 ### 缓存数据库获取的信息 ~~~ static public function getTerminalIdByTerminalMac($terminal_mac,$cache=true){ try { $result = Cache::get(__FUNCTION__ . "_{$terminal_mac}"); if ($result && $cache) { return $result; } else { $result = Db::table(OptionsCenter::$tableTerminal)->where([ OptionsCenter::$fieldTerminalMac => $terminal_mac, ])->value(OptionsCenter::$pk); if ($result) { Cache::set(__FUNCTION__ . "_{$terminal_mac}", $result, 600); return $result; } else { return ""; } } } catch (Exception $e) { Log::error($e->getMessage()); return ""; } } ~~~ ### 缓存用作全局变量 验证手机短信 以下的代码Cache的已经设置成memcached缓存 本实例方法,用缓存存储手机短信验证码 ,同时实现了避免60秒内重复发送以及验证功能 ~~~ /** * 检测手机短信验证码 * #User: Mikkle * #Email:776329498@qq.com * #Date: * @param $mobile * @param bool|false $code * @return bool */ protected function checkRegSms($mobile, $code = false) { if (!$mobile) return false; if ($code === false) { //判断60秒以内是否重复发送 if (!Cache::has('sms_' . $mobile)) return true; if (Cache::get('sms_' . $mobile)['times'] > time()) { return false; } else { return true; } } else { //判断验证码是否输入正确 if (!Cache::has('sms_' . $mobile)) return false; if (Cache::get('sms_' . $mobile)['code'] == $code) { return true; } else { return false; } } } /** * 设置手机短息验证码缓存 * #User: Mikkle * #Email:776329498@qq.com * #Date: * @param $data_cache */ protected function setRegSmsCache($data_cache) { Cache::set('sms_' . $data_cache['mobile'], $data_cache, 300); } ~~~ >[danger] 关于手机短信验证码这部分源码只是简简单单的蒙蔽同手机号码重复发送,如果真遇到恶意使用号码变换软件刷你的短信你是拦截不到的。所以自己还需要完善一下发送逻辑。我这里只是通过一个实例讲解缓存 ### 缓存用作全局变量 用于验证扫码登录 #### 微信接口端源码 ~~~ /** * 微信API接口 扫码登录 * Power by Mikkle * QQ:776329498 * @param $EventKey * @param bool|false $subscribe * @return array */ protected function messageLogEvent($EventKey,$subscribe=false){ $user_model=new PersonnelUser(); if ($user_model ->where(['we_openid'=>$this->openid])->count() ==1 ){ //写入登录缓存 $log_data=[ 'code'=>1001, 'content'=>'登录成功', 'data'=>$this->openid, ]; Cache::set('log_'.$EventKey,$log_data,120); if ($subscribe){ $reply=['type'=>'text','message'=>'感谢你的关注,你已经微信登录成功.登录ID['.$EventKey.']']; }else{ $reply=['type'=>'text','message'=>'你已经微信登录成功.登录ID['.$EventKey.']']; } }else{ $news_array=[ [ 'Title'=>'你的微信暂时未绑定ERP系统', 'Description'=>'现在去绑定', 'PicUrl'=>'', 'Url'=>$this->request->domain().'/erp/we_action/binding?sid='.$EventKey, ], ]; $reply=['type'=>'news','message'=>$news_array]; } return $reply; } ~~~ 扫码端 轮询查询接口 ~~~ /** * 扫码登录 JavaScript轮询查询接口 * Power by Mikkle * QQ:776329498 * @return array */ public function loginWe(){ $sid = $this->request->param('sid'); // if (!$data->isAjax()) return $this->showReturnCodeWithOutData(1002); if (empty($sid)) return $this->showReturnCodeWithOutData(1003); $code = [ 'code' => -1 , 'sid' => $sid, ]; if (!Cache::has('log_'.$sid) ) return $this->showReturnCodeWithOutData(1031,'数据超时失效'); $cache_data=Cache::get('log_'.$sid); if($cache_data['code']==1001){ if ($this->setOpenidLogin($sid)){ $code['code']=1001; $code['data']=$this->fans_data; } }else{ $code['code']=1033; $code['msg']='等候扫描二维码'; } return $code; } ~~~ >[info] 这里应用中 扫码端 微信接口端 协同对登录缓存值的的状态进行修改 注:二维码获取接口同样要一个缓存写入,这里的sid就是在获取二维码接口时候生成的,这里就不详细介绍了 感谢大家关注 交流请加QQ群 321449759 ![](https://box.kancloud.cn/3499008a08e64306c68873288092a057_286x340.png)