I’ve been looking for a solution to cache heavy pages and just serve a static html version for a little while now, and I’ve found a solution in output buffering.
Things to think about
It’s not a good idea to go away and cache your entire site, you need to think about which pages receive high traffic, and which pages make a number of database requests. Static HTML pages aren’t going to see a benefit from caching and may in fact be served slower due to PHP invoking the request to the cached version. as an example I’m using caching on dotdashcreate.com homepage as there are a number of database requests that could easily be cached, the cached version of the page is stored here.
If you run a big site or blog I would certainly recommend caching your homepage as this will usually be your visitors first point of contact thus generating more traffic. Its probably not a good idea to cache individual posts that allow comments etc, unless you are willing to write a script to re-cache the page.
You need to allow write access to the cache directory, in the code example this is /cache/. There’s quite a bit going on in the script, the first two lines set the path to the cached directory and the time frame to refresh the cache, we then do a check to see if the cached file is older than the cache time, if it is then it refreshes the cached version (which is the block of code at the bottom), if not it just serves the cached version.
The Code:
$cachefile = 'cache.html';
$cachetime = 4 * 60;
// Serve from the cache if it is younger than $cachetime
if (file_exists($cachefile) && time() - $cachetime < filemtime($cachefile)) {
include($cachefile);
echo "<!-- Cached copy, generated ".date('H:i', filemtime($cachefile))." -->\n";
exit;
}
ob_start(); // Start the output buffer
/* Heres where you put your page content */
// Cache the contents to a file
$cached = fopen($cacheFile, 'w');
fwrite($cached, ob_get_contents());
fclose($cached);
ob_end_flush(); // Send the output to the browser
Nice article, very well written.
I have just set this up, and it works well. For those who run expensive sql queries the same approach can be applied for some very big improvements.
Dan
Interesting post. I also use the WordPress Caching plugin for my blog. It works great. On other large ecommerce sites, I’ve often rolled my own caching backend. As an admin user changes a product, adds a new category – the caching program would kick in and rerip new static HTML / PHP pages from the dynamic PHP code I had created. One trick to pull this off is to call your dynamic PHP front-end scripts like a real user using file_get_contents. Then, store that data into a new file for customers to hit. It requires a bit of hacking to get the caching program built, but once it’s in place, you can regenerate the cached version of your site with a click.
oh.It seems very simple! thanks for sharing
Very simplify and works tutorial, Thanks.
its work-) great php tutorilas to beginers,,,thank you
very simple caching….
Great solution.. Cache-ing is difficult. If users are seeing old content, they are gone.. GJ.
Hi,
I have Any Login Form In Index Page ! Now After Login Your Code Re Generate Cache?
How To Best Chaching Page For Any Index Page With Login Form ?
Thanks
Man that’s a very good solution, and simple! Tanks for sharing!
Hey Sven good work with the guide man.
It’s a pity you don’t have a donate button! I’d most certainly donate to this outstanding blog! I guess for now i’ll settle for book-marking and adding your RSS feed to my Google account. I look forward to fresh updates and will share this site with my Facebook group. Chat soon!
I wrote I little guide where I store cache files using php and read them using apache and c++. The performance gain is really great. Read more about it:
http://sven.webiny.com/advanced-cache-mechanism-using-php-cpp-and-apache/
useful. thank you for article.
Thanks…
Pingback: Useful PHP Components & Tutorials for Everyday Project « UR-Technology
Caching is very usefull on website with big databases. I wrote a post about cache on my blog:cache file in php
Pingback: links for 2009-11-09 | Yostivanich
Thanks…
Just installed the code, works really well ! !
Nice and easy is how I like it…
Hello.
I love this tutorial but there is a problem
Why not use PHP strip slashes e.g. if i type in (with out the spaces) i could easily cross refrence……. your site
@Marc, @Steve, @ Ashley.
filectime is more accurate, so I implemented that with Marc’s advice.
Additionally, Steve is correct to use readfile. It ran considerably faster!
Thanks for this great technique!
Pingback: 40+ Invaluable PHP Tutorials and Resources | rapid-DEV.net
Idea is good, but filemtime is innacurate to give the last time the file was modified for that case. If you try it you will see.. you can modify the file and unix will tell php that the file was modified on the date it was created. Use filectime to get the last time the inode of the file was changed, since you overwrite the file each X seconds, the inode meta data will change, giving you valid informations about the file.
You might as well send HTTP cache headers as well: header(‘Cache-Control: public, max-age=’ . $cacheTime);
Also, you might get a slight perf boost using readfile() over include, since the cache file will have no PHP blocks.
Pingback: 40+ Invaluable PHP Tutorials and Resources | Ouech.net
Pingback: Angels Picks » Blog Archive » 20 Useful PHP Components & Tutorials for Everyday Project
Pingback: Dynamic page caching with PHP and output buffering | Spoonfed Project
Pingback: 20 Useful PHP Components & Tutorials for Everyday Project « Hdeya team blog
First off I have gota say that this caching idea kicks ass.
I have created a class based on the original code and user comments code. Also i fixed a couple of bugs:
here is the link:
http://www.jebaird.com/blog.php?year=2009&month=03&cat=3&id=32
Used it, in my “freshly” created cms. Works like a charm!
Thank you admin ![]()
There are some new correction too:
Replace this one:
if (isset($_SERVER['REQUEST_URI'])) // Is there any URI to cache
with this one:
if (isset($_SERVER['REQUEST_URI']) && $_SERVER['REQUEST_URI']!="/")
to escape problems when site URL ends with slash (http://site.com/)
Latest version is here:
http://paste.org/5666
P.S. Sorry for spam
Hi. Thank you for this little, but powerful code. I use SEF URL on my sites. There are problem with “/” (slash). If site URI ll be for sample – /nokia/n97/comments/page/1.html this cache script will look for cache folder, then nokia folder, then inside of nokia folder for n97 subfolder and etc… There are will be PHP errors on site (Not found page and etc.). I improved it little more.And use global variable to define is the caching this page indeed. For sample, I don’t need caching login page and etc. I use $useCache variable ($useCache=0 to disable, =1 to enable caching currently viewing page.)
And finally improved script:
http://paste.org/5665
OR
http://www.pastethat.com/use_cache_with_p
Thanks again!
P.S. Sorry for my poor English.
Great article. My homepage (index.php) is also calling the database many times. So i’ve used your approach slightly differently by creating an index.html page in the root so there is no need for the include.
nice post! i will use it
Pingback: Wordpress Blog Services - 20 Useful PHP Components & Tutorials for Everyday Project
Pingback: 20 Useful PHP Components & Tutorials for Everyday Project | Noupe
If you’re using using WordPress then the wp-supercache plugin works pretty well!
That’s a great point, I should have mentioned that in my post! I’m currently using the wp-supercache plugin, well worth installing if you anticipate spikes in blog traffic.
Designer and web developer, Co-founder and Technical Director at Harkable.com London. Previously I worked at InMobi, Spotify and MySpace. Interests include photography and making short videos. Also an avid F1 fan.
Follow us on Twitter and get in-stream updates.
Subscribe to all the Papermashup tutorials and articles straight to your RSS reader.
Sign up and get all the Papermashup tutorials straight to your inbox.
44 discussions around Caching Dynamic PHP pages easily