Laravel使用memcached缓存对文章增删改查进行优化的方法

5年以前  |  阅读数:967 次  |  编程语言:PHP 

本文实例讲述了Laravel使用memcached缓存对文章增删改查进行优化的方法。分享给大家供大家参考,具体如下:

这里我们将以文章的增删改查作为实例系统讲述缓存的使用,这个实例是对之前创建RESTFul风格控制器实现文章增删改查这篇教程的改造和升级,我们将在其基础上融合进Eloquent ORM和模型事件,将应用的场景直接拉到生成环境。

1、准备工作

路由及控制器

路由的定义和控制器的创建保持和创建RESTFul风格控制器实现文章增删改查中一样。

创建数据表

关于文章对应数据表我们在数据库部分使用查询构建器实现对数据库的高级查询已有提及,这里我们使用之前创建的数据表即可。

创建文章模型

关于文章模型Post的创建也和之前Eloquent ORM部分讲ORM概述、模型定义及基本查询中创建的一致。

2、修改控制器

在之前我们是通过缓存实现对文章的增删改查操作,这里我们将其修改为通过数据库实现增删改查操作:


    <?php
      namespace App\Http\Controllers;
      use Illuminate\Http\Request;
      use Cache;
      use App\Models\Post;
      use App\Http\Requests;
      use App\Http\Controllers\Controller;
      class PostController extends Controller
      {
        /**
         * 显示文章列表.
         *
         * @return Response
         */
        public function index()
        {
          //使用all获取所有数据,如果数据量大采用分页获取
          $posts = Post::all();
          if(!$posts)
            exit('还没有发布任何文章!');
          $html = '<ul>';
          foreach ($posts as $post) {
            $html .= '<li><a href='.route('post.show',['post'=>$post]).'>'.$post->title.'</li>';
          }
          $html .= '</ul>';
          return $html;
        }
        /**
         * 创建新文章表单页面
         *
         * @return Response
         */
        public function create()
        {
          $postUrl = route('post.store');
          $csrf_field = csrf_field();
          $html = <<<CREATE
            <form action="$postUrl" method="POST">
              $csrf_field
              <input type="text" name="title"><br/><br/>
              <textarea name="content" cols="50" rows="5"></textarea><br/><br/>
              <input type="submit" value="提交"/>
            </form>
    CREATE;
          return $html;
    }
        /**
         * 将新创建的文章存储到存储器
         *
         * @param Request $request
         * @return Response
         */
        public function store(Request $request)
        {
          $title = $request->input('title');
          $content = $request->input('content');
          $post = new Post;
          $post->title = $title;
          $post->content = $content;
          $post->save();
          return redirect()->route('post.show',['post'=>$post]);
        }
        /**
         * 显示指定文章
         *
         * @param int $id
         * @return Response
         */
        public function show($id)
        {
          $post = Cache::get('post_'.$id);
          if(!$post){
            $post = Post::find($id);
            if(!$post)
              exit('指定文章不存在!');
            Cache::put('post_'.$id,$post,60*24*7);
          }
          if(!Cache::get('post_views_'.$id))
            Cache::forever('post_views_'.$id,0);
          $views = Cache::increment('post_views_'.$id);
          Cache::forever('post_views_'.$id,$views);
          $editUrl = route('post.edit',['post'=>$post]);
          $deleteUrl = route('post.destroy',['post'=>$post]);
          $html = <<<POST
            <h3>{$post->title}</h3>
            <p>{$post->content}</p>
            <i>已有{$views}人阅读</i>
            <p>
              <a href="{$editUrl}">编辑</a>
            </p>
    POST;
          return $html;
        }
        /**
         * 显示编辑指定文章的表单页面
         *
         * @param int $id
         * @return Response
         */
        public function edit($id)
        {
          $post = Post::find($id);
          if(!$post)
            exit('指定文章不存在!');
          $postUrl = route('post.update',['post'=>$post]);
          $csrf_field = csrf_field();
          $html = <<<CREATE
            <form action="$postUrl" method="POST">
              $csrf_field
              <input type="hidden" name="_method" value="PUT"/>
              <input type="text" name="title" value="{$post->title}"><br/><br/>
              <textarea name="content" cols="50" rows="5">{$post->content}</textarea><br/><br/>
              <input type="submit" value="提交"/>
            </form>
    CREATE;
          return $html;
        }
        /**
         * 在存储器中更新指定文章
         *
         * @param Request $request
         * @param int $id
         * @return Response
         */
        public function update(Request $request, $id)
        {
          $post = Post::find($id);
          if(!$post)
            exit('指定文章不存在!');
          $title = $request->input('title');
          $content = $request->input('content');
          $post->title = $title;
          $post->content = $content;
          $post->save();
          return redirect()->route('post.show',['post'=>$post]);
        }
        /**
         * 从存储器中移除指定文章
         *
         * @param int $id
         * @return Response
         */
        public function destroy($id)
        {
          $post = Post::find($id);
          if(!$post)
            exit('指定被删除文章不存在!');
          if($post->delete()){
            redirect()->route('post.index');
          }else{
            exit('删除文章失败!');
          }
        }
      }

需要注意的是在show方法中,我们首先从缓存中取文章数据,缓存中不存在才会去数据库取,同时将数据回写到缓存中,由于对数据库的操作大部分都是读操作,所以这一点小小的改进对性能却有很大提升,尤其是在海量数据时。此外我们还将访问量持久化到缓存中以提升性能。

3、在模型事件中使用缓存

我们还可以通过模型事件在文章进行增删改的时候触发相应事件将修改保存到缓存中,这里我们简单讲模型事件注册到AppServiceProvider的boot方法中:


    //保存之后更新缓存数据
    Post::saved(function($post){
      $cacheKey = 'post_'.$post->id;
      $cacheData = Cache::get($cacheKey);
      if(!$cacheData){
        Cache::add($cacheKey,$post,60*24*7);
      }else{
        Cache::put($cacheKey,$post,60*24*7);
      }
    });
    //删除之后清除缓存数据
    Post::deleted(function($post){
      $cacheKey = 'post_'.$post->id;
      $cacheData = Cache::get($cacheKey);
      if($cacheData){
        Cache::forget($cacheKey);
      }
      if(Cache::get('post_views_'.$post->id))
        Cache::forget('post_views_'.$post->id);
    });

我们将缓存有效期设置为一周。这样在文章创建或更新时会将数据保存到缓存,而删除文章时也会从缓存中移除数据,从而保证被删除后的文章查看详情时也不能浏览。

更多关于Laravel相关内容感兴趣的读者可查看本站专题:《Laravel框架入门与进阶教程》、《php优秀开发框架总结》、《smarty模板入门基础教程》、《php日期与时间用法总结》、《php面向对象程序设计入门教程》、《php字符串(string)用法总结》、《php+mysql数据库操作入门教程》及《php常见数据库操作技巧汇总

希望本文所述对大家基于Laravel框架的PHP程序设计有所帮助。

 相关文章:
PHP分页显示制作详细讲解
SSH 登录失败:Host key verification failed
获取IMSI
将二进制数据转为16进制以便显示
获取IMEI
文件下载
贪吃蛇
双位运算符
PHP自定义函数获取搜索引擎来源关键字的方法
Java生成UUID
发送邮件
年的日历图
提取后缀名
在Zeus Web Server中安装PHP语言支持
让你成为最历害的git提交人
Yii2汉字转拼音类的实例代码
再谈PHP中单双引号的区别详解
指定应用ID以获取对应的应用名称
Python 2与Python 3版本和编码的对比
php封装的page分页类完整实例