在项目中,合理的使用缓存对性能有较大的帮助。ThinkPHP提供了方便的缓存方式,包括数据缓存、静态缓存和查询缓存等,支持包括文件方式、APC、Db、Memcache、Shmop、Sqlite、Redis、Eaccelerator和Xcache在内的动态数据缓存类型,以及可定制的静态缓存规则,并提供了快捷方法进行存取操作。
# 数据缓存
在ThinkPHP中进行缓存操作,一般情况下并不需要直接操作缓存类,因为系统内置对缓存操作进行了封装,直接采用S方法即可,例如:
## 缓存初始化
~~~
// 缓存初始化
S(array('type'=>'xcache','expire'=>60));
~~~
缓存初始化可以支持的参数根据不同的缓存方式有所区别,常用的参数是:
| 参数 | 描述 |
|-----|-----|
| expire | 缓存有效期(时间为秒) |
| prefix | 缓存标识前缀 |
| type | 缓存类型 |
系统目前已经支持的缓存类型包括:`Apachenote、Apc、Db、Eaccelerator、File、Memcache、Redis、Shmop、Sqlite、Wincache和Xcache`。
如果S方法不传入type参数初始化的话,则读取配置文件中设置的`DATA_CACHE_TYPE`参数值作为默认类型。同样的道理,prefix参数如果没有传入会读取配置文件的`DATA_CACHE_PREFIX`参数值,expire参数没有传入则读取`DATA_CACHE_TIME`配置值作为默认。
有些缓存方式会有一些自身特殊的参数,例如Memcache缓存,还需要配置其他的参数:
~~~
S(array(
'type'=>'memcache',
'host'=>'192.168.1.10',
'port'=>'11211',
'prefix'=>'think',
'expire'=>60)
);
~~~
对于全局的缓存方式,一般我们建议添加prefix(缓存前缀)参数用以区分不同的应用,以免混淆。
## 缓存设置
~~~
// 设置缓存
S('name',$value);
~~~
会按照缓存初始化时候的参数进行缓存数据,也可以在缓存设置的时候改变参数,例如:
~~~
// 缓存数据300秒
S('name',$value,300);
~~~
甚至改变之前的缓存方式或者更多的参数:
~~~
// 采用文件方式缓存数据300秒
S('name',$value,array('type'=>'file','expire'=>300));
~~~
如果你在缓存设置的时候采用上面的数组方式传入参数的话,会影响到后面的缓存存取。
## 缓存读取
~~~
// 读取缓存
$value = S('name');
~~~
缓存读取的是前面缓存设置的值,这个值会受缓存初始化或者缓存设置的时候传入的参数影响。 如果缓存标识不存在或者已经过期,则返回false,否则返回缓存值。
## 缓存删除
~~~
// 删除缓存
S('name',null);
~~~
删除缓存标识为name的缓存数据。
## 对象方式操作缓存
我们可以采用对象方式操作缓存,例如:
~~~
// 初始化缓存
$cache = S(array('type'=>'xcache','prefix'=>'think','expire'=>600));
$cache->name = 'value'; // 设置缓存
$value = $cache->name; // 获取缓存
unset($cache->name); // 删除缓存
~~~
> 如果你设置了缓存前缀的话,对应的缓存操作只是对应该缓存前缀标识的,不会影响其他的缓存。
## 关于文件缓存方式的安全机制
如果你使用的是文件方式的缓存机制,那么可以设置DATA_CACHE_KEY参数,避免缓存文件名被猜测到,例如:
~~~
'DATA_CACHE_KEY'=>'think'
~~~
## 缓存队列
数据缓存可以支持缓存队列,简单的说就是可以限制缓存的数量,只需要在初始化的时候指定`length`参数:
~~~
S(array('type'=>'xcache','length'=>100,'expire'=>60));
~~~
设置了`length`参数后,系统只会缓存最近的100条缓存数据。
# 快速缓存
如果你的存储数据没有有效期的需求,那么系统还提供了一个快速缓存方法F可以用来更快的操作。
> F方法可以支持不同的存储类型,如果是文件类型的话,默认保存在DATA_PATH目录下面。
快速缓存Data数据
~~~
F('data',$Data);
~~~
快速缓存Data数据,保存到指定的目录
~~~
F('data',$Data,TEMP_PATH);
~~~
获取缓存数据
~~~
$Data = F('data');
~~~
删除缓存数据
~~~
F('data',NULL);
~~~
F方法支持自动创建缓存子目录,在`DATA_PATH`目录下面缓存data数据,如果User子目录不存在,则自动创建:
~~~
F('User/data',$Data);
~~~
系统内置的数据字段信息缓存就是用了快速缓存机制。
# 查询缓存
对于及时性要求不高的数据查询,我们可以使用查询缓存功能来提高性能,而且无需自己使用缓存方法进行缓存和获取。
查询缓存功能支持所有的数据库,并且支持所有的缓存方式和有效期。
在使用查询缓存的时候,只需要调用Model类的cache方法,例如:
~~~
$Model->cache(true)->where('status=1')->select();
~~~
如果使用了`cache(true)` ,则在查询的同时会根据当前的查询条件等信息生成一个带有唯一标识的查询缓存,如果指定了key的话,则直接生成名称为key的查询缓存 ,例如:
~~~
$Model->cache('cache_name')->select();
~~~
> 指定key的方式会让查询缓存更加高效。
默认情况下缓存方式采用**DATA_CACHE_TYPE**参数设置的缓存方式(系统默认值为File表示采用文件方式缓存),缓存有效期是**DATA_CACHE_TIME**参数设置的时间,也可以单独制定查询缓存的缓存方式和有效期:
~~~
$Model->cache(true,60,'xcache')->select();
~~~
表示当前查询缓存的缓存方式为xcache,并且缓存有效期为60秒。
同样的查询,如果没有使用cache方法,则不会获取或者生成任何缓存,即便是之前调用过Cache方法。
如果指定了查询缓存的key的话,则可以在外部通过S方法直接获取查询缓存的内容,例如:
~~~
$value = S('cache_name');
~~~
除了select方法之外,查询缓存还支持find和getField方法,以及他们的衍生方法(包括统计查询和动态查询方法)。
~~~
// 对查询数据缓存60秒
$Model->where($map)->cache('key',60)->find();
~~~
具体应用的时候可以根据需要选择缓存方式和缓存有效期。
# SQL解析缓存
除了查询缓存之外,ThinkPHP还支持SQL解析缓存,因为ThinkPHP的ORM机制,所有的SQL都是动态生成的,然后由数据库驱动执行。
所以如果你的应用有大量的SQL查询需求,那么可以开启SQL解析缓存以减少SQL解析提高性能。要开启SQL解析缓存,只需要设置:
~~~
'DB_SQL_BUILD_CACHE' => true,
~~~
即可开启数据库查询的SQL创建缓存,默认缓存方式为文件方式,还可以支持xcache和apc方式缓存,只需要设置:
~~~
'DB_SQL_BUILD_QUEUE' => 'xcache',
~~~
我们知道,一个项目的查询SQL的量可能会非常巨大,所以有必要设置下缓存的队列长度,例如,我们希望SQL解析缓存不超过20条记录,可以设置:
~~~
'DB_SQL_BUILD_LENGTH' => 20, // SQL缓存的队列长度
~~~
> 注意:只有查询方法才支持SQL解析缓存
# 静态缓存
要使用静态缓存功能,需要开启`HTML_CACHE_ON`参数,并且使用`HTML_CACHE_RULES`配置参数设置静态缓存规则文件 。
> 虽然也可以在应用配置文件中定义静态缓存规则,但是建议是在模块配置文件中为不同的模块定义静态缓存规则。
## 静态规则定义
静态规则的定义方式如下:
~~~
'HTML_CACHE_ON' => true, // 开启静态缓存
'HTML_CACHE_TIME' => 60, // 全局静态缓存有效期(秒)
'HTML_FILE_SUFFIX' => '.shtml', // 设置静态缓存文件后缀
'HTML_CACHE_RULES' => array( // 定义静态缓存规则
// 定义格式1 数组方式
'静态地址' => array('静态规则', '有效期', '附加规则'),
// 定义格式2 字符串方式
'静态地址' => '静态规则',
)
~~~
定义格式1采用数组方式 便于单独为某个静态规则设置不同的有效期,定义格式2采用字符串方式订阅静态规则,同时采用`HTML_CACHE_TIME`设置的全局静态缓存有效期。
> 静态缓存文件的根目录在`HTML_PATH`定义的路径下面,并且只有定义了静态规则的操作才会进行静态缓存。 并且静态缓存支持不同的存储类型。 静态缓存仅在GET请求下面有效。
### 静态地址
静态地址包括下面几种定义格式:
第一种是定义全局的操作静态规则,例如定义所有的read操作的静态规则为:
~~~
'read'=>array('{id}',60)
~~~
其中,`{id}` 表示取`$_GET['id']` 为静态缓存文件名,第二个参数表示缓存60秒。
第二种是定义全局的控制器静态规则,例如定义所有的User控制器的静态规则为:
~~~
'user:'=>array('User/{:action}_{id}','600')
~~~
其中,`{:action}` 表示当前的操作名称
第三种是定义某个控制器的操作的静态规则,例如,我们需要定义Blog控制器的read操作进行静态缓存
~~~
'blog:read'=>array('{id}',0)
~~~
第四种方式是定义全局的静态缓存规则,这个属于特殊情况下的使用,任何模块的操作都适用,例如
~~~
'*'=>array('{$_SERVER.REQUEST_URI|md5}'),
~~~
表示根据当前的URL进行缓存。
### 静态规则
静态规则是用于定义要生成的静态文件的名称,静态规则的定义要确保不会冲突,写法可以包括以下情况:
### 1、使用系统变量
包括 `_GET`、`_REQUEST`、`_SERVER`、`_SESSION`、`_COOKIE` 格式:
~~~
{$_×××|function}
~~~
例如:
~~~
{$_GET.name}
{$_SERVER.REQUEST_URI|md5}
~~~
### 2、使用框架特定的变量
`{:module}` 、`{:controller}` 和`{:action}` 分别表示当前模块名、控制器名和操作名。
例如:
~~~
{:module}/{:controller}_{:action}
~~~
### 3、使用_GET变量
`{var|function}`也就是说 `{id}` 其实等效于 `{$_GET.id}`
#### 4、直接使用函数
{|function} 例如:
~~~
{|time}
~~~
### 5、支持混合定义
例如我们可以定义一个静态规则为:
~~~
'{id},{name|md5}'
~~~
在{}之外的字符作为字符串对待,如果包含有"/",会自动创建目录。
例如,定义下面的静态规则:
~~~
{:module}/{:action}_{id}
~~~
则会在静态目录下面创建模块名称的子目录,然后写入操作名_id.shtml 文件。
### 静态缓存有效期
单位为秒。如果不定义,则会获取配置参数`HTML_CACHE_TIME` 的设置值,如果定义为0则表示永久缓存。
### 附加规则
通常用于对静态规则进行函数运算,例如
~~~
'read'=>array('Think{id},{name}','60', 'md5')
~~~
翻译后的静态规则是 `md5('Think'.$_GET['id']. ', '.$_GET['name']);`
- 序
- 前言
- 内容简介
- 目录
- 基础知识
- 起步
- 控制器
- 模型
- 模板
- 命名空间
- 进阶知识
- 路由
- 配置
- 缓存
- 权限
- 扩展
- 国际化
- 安全
- 单元测试
- 拿来主义
- 调试方法
- 调试的步骤
- 调试工具
- 显示trace信息
- 开启调试和关闭调试的区别
- netbeans+xdebug
- Socketlog
- PHP常见错误
- 小黄鸭调试法,每个程序员都要知道的
- 应用场景
- 第三方登录
- 图片处理
- 博客
- SAE
- REST实践
- Cli
- ajax分页
- barcode条形码
- excel
- 发邮件
- 汉字转全拼和首字母,支持带声调
- 中文分词
- 浏览器useragent解析
- freelog项目实战
- 需求分析
- 数据库设计
- 编码实践
- 前端实现
- rest接口
- 文章发布
- 文件上传
- 视频播放
- 音乐播放
- 图片幻灯片展示
- 注册和登录
- 个人资料更新
- 第三方登录的使用
- 后台
- 微信的开发
- 首页及个人主页
- 列表
- 归档
- 搜索
- 分页
- 总结经验
- 自我提升
- 进行小项目的锻炼
- 对现有轮子的重构和移植
- 写技术博客
- 制作视频教程
- 学习PHP的知识和新特性
- 和同行直接沟通、交流
- 学好英语,走向国际
- 如何参与
- 浏览官网和极思维还有看云
- 回答ThinkPHP新手的问题
- 尝试发现ThinkPHP的bug,告诉官方人员或者push request
- 开发能提高效率的ThinkPHP工具
- 尝试翻译官方文档
- 帮新手入门
- 创造基于ThinkPHP的产品,进行连带推广
- 展望未来
- OneThink
- ThinkPHP4
- 附录