<?php

namespace App\Services;

use Illuminate\Support\Facades\Log;
use Illuminate\Support\Facades\Redis; class CacheDecorator
{
// 公司组织列表缓存key
const COMPANY_ORGANIZE_KEY = 'company_organize_%s'; private static function setStringCacheStorage($key, $expireTime, string $data)
{
if ($expireTime) {
Redis::setex($key, $expireTime, $data);
} else {
Redis::set($key, $data);
}
} /**
* 装饰器获取缓存数据
* @param $cacheKey 缓存key
* @param int $cacheTime 缓存时间
* @param object $obj 对象
* @param $method 缓存方法
* @param mixed ...$args 参数
* @return mixed
*/
public static function getStringCache($cacheKey, $expireTime, object $obj, $method, ...$args)
{
try {
$cacheData = Redis::get($cacheKey);
$cacheData = json_decode($cacheData, true);
if (empty($cacheData) || !isset($cacheData['org_data']) || empty($cacheData['org_data'])) {
$cacheData = [
'org_data' => call_user_func_array([$obj, $method], $args)
];
self::setStringCacheStorage($cacheKey, $expireTime, json_encode($cacheData, JSON_UNESCAPED_UNICODE));
Log::info('从数据库获取数据', [
'class' => get_class($obj),
'method' => $method,
'args' => $args,
]);
return $cacheData['org_data'];
} else {
Log::info('从缓存获取数据', [
'class' => get_class($obj),
'method' => $method,
'args' => $args,
]);
return $cacheData['org_data'];
}
} catch (\Exception $e) {
Log::error("redis缓存错误", [
'msg' => $e->getMessage(),
'line' => $e->getLine(),
'file' => $e->getFile(),
]);
}
Log::info('从数据库获取数据', [
'class' => get_class($obj),
'method' => $method,
'args' => $args,
]);
return call_user_func_array([$obj, $method], $args);
} /**
* 获取缓存key
* @param $format
* @param mixed ...$arg
* @return string
*/
public static function getCacheKey($format, ...$arg)
{
return sprintf($format, ...$arg);
}
}

调用

   public function listOrganize(AuthOrganizeService $authOrganizeService)
{
// before直接调用数据库
$list = $authOrganizeService->listCompanyOrganize(Auth::user()->company_id);
// after 直接使用缓存,但是listCompanyOrganize这个方法是没有进行任何的改动
$list = CacheDecorator::getStringCache(
CacheDecorator::getCacheKey(CacheDecorator::COMPANY_ORGANIZE_KEY),
100,
AuthOrganizeService,
'listCompanyOrganize',
1
);
$res = [
'list' => $list,
'total' => 1 //不分页
];
return $this->success($res);
}

php使用装饰模式无侵入式加缓存的更多相关文章

  1. Hook 无侵入式埋点(页面统计)

    一.技术原理 Method-Swizzling 黑魔法 方法交换(不懂的可以查) 二.页面统计 某盟页面统计SDK需要开发者在APP基类里实现ViewDidAppear和viewDidDisappea ...

  2. Spring Boot 无侵入式 实现RESTful API接口统一JSON格式返回

    前言 现在我们做项目基本上中大型项目都是选择前后端分离,前后端分离已经成了一个趋势了,所以总这样·我们就要和前端约定统一的api 接口返回json 格式, 这样我们需要封装一个统一通用全局 模版api ...

  3. 使用AOP思想无侵入式申请权限,解决组件化中权限问题(一)

    首先介绍AspectJx使用 https://github.com/HujiangTechnology/gradle_plugin_android_aspectjx 在根项目的build.gradle ...

  4. Android平台免Root无侵入AOP框架Dexposed使用详解

    Dexposed是基于久负盛名的开源Xposed框架实现的一个Android平台上功能强大的无侵入式运行时AOP框架. Dexposed的AOP实现是完全非侵入式的,没有使用任何注解处理器,编织器或者 ...

  5. Android新技术学习——阿里巴巴免Root无侵入AOP框架Dexposed

    阿里巴巴无线事业部近期开源的Android平台下的无侵入运行期AOP框架Dexposed,该框架基于AOP思想,支持经典的AOP使用场景.可应用于日志记录,性能统计,安全控制.事务处理.异常处理等方面 ...

  6. 小议webpack下的AOP式无侵入注入

    说起来, 面向切面编程(AOP)自从诞生之日起,一直都是计算机科学领域十分热门的话题,但是很奇怪的是,在前端圈子里,探讨AOP的文章似乎并不是多,而且多数拘泥在给出理论,然后实现个片段的定式)难免陷入 ...

  7. MVC的验证(模型注解和非侵入式脚本的结合使用) .Net中初探Redis .net通过代码发送邮件 Log4net (Log for .net) 使用GDI技术创建ASP.NET验证码 Razor模板引擎 (RazorEngine) .Net程序员应该掌握的正则表达式

    MVC的验证(模型注解和非侵入式脚本的结合使用)   @HtmlHrlper方式创建的标签,会自动生成一些属性,其中一些属性就是关于验证 如图示例: 模型注解 通过模型注解后,MVC的验证,包括前台客 ...

  8. Android沉浸式(侵入式)标题栏(状态栏)Status(一)

     Android沉浸式(侵入式)标题栏(状态栏)Status(一) 现在越来越多的APP设计采用这种称之为沉浸式状态栏(Status)的设计,这种沉浸式状态栏又称之"侵入式"状 ...

  9. 无侵入方面编程-用HttpModule+SoapExtension监视页面执行参数(二)

    上一篇文章 "无侵入方面编程-用HttpModule+SoapExtension监视页面执行参数(一)"中,我们实现了监视每个页面的执行情况和调用WebService的简单信息. ...

随机推荐

  1. mybatis整合Spring编码

    mybatis整合Spring的核心代码 spring-dao.xml <?xml version="1.0" encoding="UTF-8"?> ...

  2. org.apache.ibatis.binding.BindingException: Parameter '0' not found. Available parameters are [arg1, arg0, param1, param2]

    报错信息如下: org.apache.ibatis.binding.BindingException: Parameter '0' not found. Available parameters ar ...

  3. js栈和堆的区别

    一.  堆(heap)和栈(stack) 栈(stack)会自动分配内存空间,会自动释放.堆(heap)动态分配的内存,大小不定也不会自动释放. 二.  基本类型和引用类型 基本类型:简单的数据段,存 ...

  4. [Python之路] 元类(引申 单例模式)

    一.类也是对象 当我们定义一个变量或者函数的时候,我们可以在globals()的返回值字典中找到响应的映射: def A(): print("This is function A" ...

  5. 主流包管理工具npm、yarn、cnpm、pnpm之间的区别与联系——原理篇

    接触 node 之后,一直使用npm包管理工具, cnpm 一开始会用一些,但是并没有觉得比 npm 快得多,使用 cnpm 的时候还经常安装不成功,只能再用 npm 安装一遍,渐渐的就弃用了 cnp ...

  6. ftell函数

    ftell函数用于得到文件位置指针当前位置相对于文件首的偏移字节数,在随机方式存储文件时,由于文件位置频繁的前后移动,程序不容易确定文件的当前位置. /*** a.txt ***/ asd gsder ...

  7. MySQL_(Java)使用JDBC向数据库发起查询请求

    MySQL_(Java)使用JDBC向数据库发起查询请求 传送门 MySQL_(Java)使用JDBC创建用户名和密码校验查询方法 传送门 MySQL_(Java)使用preparestatement ...

  8. 【Python】模块学习之locust性能测试

    背景 locust是一个python的第三方库,用于做性能测试,可使用多台机器同时对一台服务器进行压测,使用其中一台机器作为主节点,进行分布式管理 博主测试接口的时候一直是使用python脚本,后来有 ...

  9. IP输出 之 分片ip_fragment、ip_do_fragment

    概述 ip_fragment函数用于判断是否进行分片,在没有设置DF标记的情况下进入分片,如果设置了DF标记,则继续判断,如果不允许DF分片或者收到的最大分片大于MTU大小,则回复ICMP,释放skb ...

  10. 浏览器缓存及vw和vh的使用

    在浏览器缓存中不仅有 cookie 还有了别的选择 Storage 浏览器又分了两种缓存:sessionStorage localStorage localStorage 缓存:是一种永久的缓存,也就 ...