首页 > 代码库 > CodeIgniter框架的缓存原理分解
CodeIgniter框架的缓存原理分解
用缓存的目的:(手册上叙述如下,已经写得很清楚了)
Codeigniter 支持缓存技术,以达到最快的速度。
尽管CI已经相当高效了,但是网页中的动态内容、主机的内存CPU 和数据库读取速度等因素直接影响了网页的加载速度。 依靠网页缓存,你的网页可以达到近乎静态网页的加载速度,因为他们将程序输出的结果保存到硬盘上了。
如何用:
启用缓存功能,只需要将下面的代码放入你的任何一个控制器(controller)的方法(function)内:
$this->output->cache(1);
其中 n 是你希望缓存更新的 分钟 数。可以使用 m/60 来精确到秒,例如 1/60 ,则是精确到 1秒
上面的代码可以放到任何一个 function 里面。他的出现顺序对缓存并没有影响,所以将它放在你认为最合乎逻辑的地方。一旦上面的代码放到了控制器的方法中,页面就会被缓存。
注意:
由于CI存储缓存文件的方式,只有通过 view 文件的输出才能被缓存,也即是$this->load->view(‘xxxx‘, $data); 第三个参数不能为true
在缓存文件产生之前,请确保 application/cache 文件夹可写。
如果你不再想使用缓存,仅需将上面的代码从你的controller里面删除即可。注意: 这样做并不能让缓存文件立即消失,它将会自动过期并被删除。如果你想立即删除那些文件,就必须自己动手了。
原理:
使用$this->output->cache(1); 开启缓存:
然后是CodeIgniter.php中做判断:
1 if ($EXT->_call_hook(‘cache_override‘) === FALSE)2 {3 if ($OUT->_display_cache($CFG, $URI) == TRUE)4 {5 exit;6 }7 }
上面第三行代码,如果读取缓存文件成功,_display_cache方法直接输出缓存内容,然后exit掉不再往下执行
读缓存操作:(Output Class _display_cache方法里边)
读缓存文件,没有返回false
有缓存文件 判断缓存是否过期,过期则删除缓存文件返回false
没有过期,读取缓存内容显示 返回false
* Update/serve a cached file * * @access public * @param object config class * @param object uri class * @return void */ function _display_cache(&$CFG, &$URI) { $cache_path = ($CFG->item(‘cache_path‘) == ‘‘) ? APPPATH.‘cache/‘ : $CFG->item(‘cache_path‘); // Build the file path. The file name is an MD5 hash of the full URI $uri = $CFG->item(‘base_url‘). $CFG->item(‘index_page‘). $URI->uri_string; $filepath = $cache_path.md5($uri); if ( ! @file_exists($filepath)) { return FALSE; } if ( ! $fp = @fopen($filepath, FOPEN_READ)) { return FALSE; } flock($fp, LOCK_SH); $cache = ‘‘; if (filesize($filepath) > 0) { $cache = fread($fp, filesize($filepath)); } flock($fp, LOCK_UN); fclose($fp); // Strip out the embedded timestamp if ( ! preg_match("/(\d+TS--->)/", $cache, $match)) { return FALSE; } // Has the file expired? If so we‘ll delete it. if (time() >= trim(str_replace(‘TS--->‘, ‘‘, $match[‘1‘]))) { if (is_really_writable($cache_path)) { @unlink($filepath); log_message(‘debug‘, "Cache file has expired. File deleted"); return FALSE; } } // Display the cache $this->_display(str_replace($match[‘0‘], ‘‘, $cache)); log_message(‘debug‘, "Cache file is current. Sending it to browser."); return TRUE; }
写缓存操作:(Output Class _write_cache方法里边)
缓存文件名称: URI 的md5值;
缓存内容:过期时间戳加特殊字符‘TS--->‘ 再加变量解析后的模板内容
缓存路径可自定义
function _write_cache($output) { $CI =& get_instance(); $path = $CI->config->item(‘cache_path‘); $cache_path = ($path == ‘‘) ? APPPATH.‘cache/‘ : $path; if ( ! is_dir($cache_path) OR ! is_really_writable($cache_path)) { log_message(‘error‘, "Unable to write cache file: ".$cache_path); return; } $uri = $CI->config->item(‘base_url‘). $CI->config->item(‘index_page‘). $CI->uri->uri_string(); $cache_path .= md5($uri); if ( ! $fp = @fopen($cache_path, FOPEN_WRITE_CREATE_DESTRUCTIVE)) { log_message(‘error‘, "Unable to write cache file: ".$cache_path); return; } $expire = time() + ($this->cache_expiration * 60); if (flock($fp, LOCK_EX)) { fwrite($fp, $expire.‘TS--->‘.$output); flock($fp, LOCK_UN); } else { log_message(‘error‘, "Unable to secure a file lock for file at: ".$cache_path); return; } fclose($fp); @chmod($cache_path, FILE_WRITE_MODE); log_message(‘debug‘, "Cache file written: ".$cache_path); }