ThinkPHP缓存源码深度解析(1)

本文涉及的产品
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: ThinkPHP缓存源码深度解析

前言

在项目中缓存是必不可少的一项功能,当用户量大的时候是必须上缓存的,如何都直接查数据库那么对于用户体验来说就太差了。


那么什么情况下应该使用缓存呢!


  • 热点事件,例如微博热搜
  • 不经常更新的数据,例如配置项
  • 博客平台的排行榜单
  • 社交平台的关注列表粉丝列表等等

以上说的这些应用场景并不说是框架的缓存,一般在使用缓存的层面是不太使用框架的缓存的。


常用的为redis,memcache等NoSQL。


但是今天主要讨论的是框架中缓存,所以千万不要认为框架的缓存是无所不能的,还是要看项目的实际情况。


一、缓存cache设置的执行流程以及源码解析

首先需要实现以下的案例,并且引入cache类


image.png


cache怎么运行的?


就代码Cache::set这个现在知道是怎么运行的吗?如果不知道咔咔带你在来深入的学习一次。


我们都知道框架的入口文件是index.php,在入口文件中引入了一个文件为base.php。


image.png


来到base.php这个文件里边可以看到关于注册类库别名,至于是怎么注册的,这个在框架执行流程的那一节中有过深度的讲解,可以回过头在去了解一下。


image.png


所以说代码将会执行到框架核心的facade这个类里边,在这个类里边存在一个方法__callStatic,当调用不存在的静态方法时此方法会进行执行。


image.png


那么怎么来做这个验证呢!不能咔咔这样说就是这样的对吧!


那么代码将会接着来到创建Facade实例这个方法,我们做的测试就是将这个class打印出来得到的值都有什么。


image.png


暂时先不管这个cache执行了几次,是可以明显的看到打印结果是存在这个值的,所以说从另一个笨拙的方面验证了咔咔的说辞。


image.png


这里有一个特别小的细节我想大家应该需要了解一下,那就是关于static的使用


关于static的小技巧


首先可以看到cache类是继承这Facade门面类


image.png


然后static是在门面类中做的使用,那么最终返回的类就是继承门面类的那个类也就是cache类


image.png


总结为一句话就为


static 如果有被继承的话 默认调用子类 ,否则调用的是自身


所以说下边接着的static::getFacadeClass()这里也是执行的子类中的方法。


好了,进入了一段小插曲,接下来会到正题。


所以说代码将会执行到thinkphp/library/think/Cache.php这个文件,也就是核心类库的位置。


在这个方法你是找不到set方法的,所以代码将会执行到__call方法,这个方法当调用不存在的方法时则会触发的方法。


image.png


自动初始化缓存


根据执行流程我们将会看到init这个方法自动初始化缓存(这里需要注意,第一次并不是在这里进行执行的,而是make方法,当make方法执行完后会把值存放在handler这个属性,第二次通过call方法进来之后就直接返回了,而不会在进行一次执行,这里一定要注意)


image.png


在这里我们进行打印一次$options这个的值。


image.png


探讨一下为什么$options这个参数会有值


这里就是关于容器方面的知识了,来咔咔带你看一下。


image.png


当在创建Cache时创建Facade实例,在这个过程中注意咔咔下图圈起来的部位,执行了一个见了八百次的make方法了。


image.png


来到make方法只需要看咔咔圈起来的地方即可


image.png


然后在进入到invokeClass方法,这个方法是调用反射执行类的实例化 支持依赖注入。


在这个方法中通过反射执行了Cache中的make方法。


image.png


所以就会执行Cache类中的make方法,这个方法就会进行实例化本类,并且执行构造函数,接下来看一下。


image.png


来到构造函数中你会看到从make方法获取到的cache配置文件的配置项传进了init方法,也就是自动初始化缓存的部分。


image.png


所以从这里看到init方法自动初始化缓存第一次执行是在容器实例化的时候执行的所以$options才会存在值。


接下来将顺着这个流程进行连接缓存也就是代码$this->handler = $this->connect($options);这块的内容。


这个方法就很简单了,就是使用了之前一直讲解的工厂模式实现的加载不同类型的缓存方式。


然后会把返回的对象存放在以$optionsmd5为下标的缓存实例属性$instance里边。


image.png


最终代码会返回给cache中的__call方法,类为object(think\cache\driver\File)方法为set


image.png


于是执行流程会来到下图位置写入缓存


image.png


获取文件名


在这个方法中主要需要理解的一件事情就是在缓存中是如何进行获取具体的文件名然后进行存储数据的。


这个name值就是咱们需要设置的值,wechat。


image.png


然后来到getCacheKey取得变量的存储文件名。


在这个方法中第一步就是通过hash的方式进行类型和缓存值加密,这个options是在本类声明好了的,这里一定要明确。


因为在框架中大量的使用了options这个变量千万不要搞混淆了。


在这个方法中需要明白的就是这个文件名是怎么确定的。


image.png


还是要来到本类的开始位置查看一下这个options的值,在这个类中可以看到上图中使用的加密类型为hash_type就是md5


image.png


然后来到构造函数中可以看到关于path的设置


image.png


在上图中可以看到Container::get('app')这行代码,这行代码就是使用的容器执行的也是make方法,关于这个make方法在容器中起到的作用是十分大的,所以需要好好理解。


然后有一个小细节不知道大家有没有看到,那就是在下方有一个init方法,我们一起去看看这个方法是干什么的。


来到这个方法后你会发现这里是直接按照获取到的缓存文件的路径进行创建文件。


image.png


这时可以查看一下创建的文件,可以看到文件已经创建好了。


image.png


最后通过file_put_contents函数将数据存放至刚刚获取到的缓存文件存放位置


image.png


数据库存储形式就是下图


image.png


直到这里关于框架缓存设置就结束了,其实流程并不难,在这个案例中咔咔使用的文件形式的,至于redis还是其它都是一样的。



相关文章
|
11天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
39 2
|
11天前
|
存储 缓存 网络协议
如何防止DNS缓存中毒攻击(一)
DNS缓存中毒也称为DNS欺骗
40 10
|
11天前
|
缓存 网络协议 安全
如何防止DNS缓存中毒(Ⅱ)
服务器应该配置为尽可能少地依赖与其他DNS服务器的信任关系
33 10
|
11天前
|
存储 安全 Linux
Golang的GMP调度模型与源码解析
【11月更文挑战第11天】GMP 调度模型是 Go 语言运行时系统的核心部分,用于高效管理和调度大量协程(goroutine)。它通过少量的操作系统线程(M)和逻辑处理器(P)来调度大量的轻量级协程(G),从而实现高性能的并发处理。GMP 模型通过本地队列和全局队列来减少锁竞争,提高调度效率。在 Go 源码中,`runtime.h` 文件定义了关键数据结构,`schedule()` 和 `findrunnable()` 函数实现了核心调度逻辑。通过深入研究 GMP 模型,可以更好地理解 Go 语言的并发机制。
|
24天前
|
消息中间件 缓存 安全
Future与FutureTask源码解析,接口阻塞问题及解决方案
【11月更文挑战第5天】在Java开发中,多线程编程是提高系统并发性能和资源利用率的重要手段。然而,多线程编程也带来了诸如线程安全、死锁、接口阻塞等一系列复杂问题。本文将深度剖析多线程优化技巧、Future与FutureTask的源码、接口阻塞问题及解决方案,并通过具体业务场景和Java代码示例进行实战演示。
40 3
|
25天前
|
缓存 网络协议 安全
如何防止DNS缓存中毒(Ⅱ)
防止DNS缓存中毒的方法包括:减少DNS服务器与其它服务器的信任关系;限制DNS服务器上的服务;使用最新版DNS;加强用户安全教育,如识别可疑网站,仅访问HTTPS网站等。部署SSL证书并选择符合国际Webtrust标准的CA机构,可进一步提高安全性。
37 1
|
1月前
|
存储
让星星⭐月亮告诉你,HashMap的put方法源码解析及其中两种会触发扩容的场景(足够详尽,有问题欢迎指正~)
`HashMap`的`put`方法通过调用`putVal`实现,主要涉及两个场景下的扩容操作:1. 初始化时,链表数组的初始容量设为16,阈值设为12;2. 当存储的元素个数超过阈值时,链表数组的容量和阈值均翻倍。`putVal`方法处理键值对的插入,包括链表和红黑树的转换,确保高效的数据存取。
56 5
|
25天前
|
存储 缓存 网络协议
如何防止DNS缓存中毒攻击(一)
DNS缓存中毒,即DNS欺骗,是一种通过利用DNS系统的漏洞,将用户流量从合法服务器导向虚假服务器的网络攻击。攻击者通过伪造DNS响应,使缓存服务器存储错误的IP地址,从而实现对合法URL的劫持。这不仅可能导致用户信息泄露,还可能使用户设备遭受恶意软件感染,对金融、医疗等关键领域造成严重影响。据统计,DNS攻击每年造成的平均损失高达223.6万美元,其中23%的攻击源自DNS缓存中毒。
59 0
|
1月前
|
消息中间件 缓存 NoSQL
Redis 是一个高性能的键值对存储系统,常用于缓存、消息队列和会话管理等场景。
【10月更文挑战第4天】Redis 是一个高性能的键值对存储系统,常用于缓存、消息队列和会话管理等场景。随着数据增长,有时需要将 Redis 数据导出以进行分析、备份或迁移。本文详细介绍几种导出方法:1)使用 Redis 命令与重定向;2)利用 Redis 的 RDB 和 AOF 持久化功能;3)借助第三方工具如 `redis-dump`。每种方法均附有示例代码,帮助你轻松完成数据导出任务。无论数据量大小,总有一款适合你。
77 6
|
14天前
|
缓存 NoSQL 关系型数据库
大厂面试高频:如何解决Redis缓存雪崩、缓存穿透、缓存并发等5大难题
本文详解缓存雪崩、缓存穿透、缓存并发及缓存预热等问题,提供高可用解决方案,帮助你在大厂面试和实际工作中应对这些常见并发场景。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:如何解决Redis缓存雪崩、缓存穿透、缓存并发等5大难题
下一篇
无影云桌面