Chapitre 4. Zend_Cache

Table des matières

4.1. Introduction
4.2. La théorie du cache
4.2.1. La méthode de fabrique de Zend_Cache
4.2.2. Baliser les enregistrements
4.2.3. Nettoyer le cache
4.3. Frontends Zend_Cache
4.3.1. Zend_Cache_Core
4.3.2. Zend_Cache_Frontend_Output
4.3.3. Zend_Cache_Frontend_Function
4.3.4. Zend_Cache_Frontend_Class
4.3.5. Zend_Cache_Frontend_File
4.3.6. Zend_Cache_Frontend_Page
4.4. Zend_Cache backends
4.4.1. Zend_Cache_Backend_File
4.4.2. Zend_Cache_Backend_Sqlite
4.4.3. Zend_Cache_Backend_Memcached
4.4.4. Zend_Cache_Backend_APC
4.4.5. Zend_Cache_Backend_ZendPlatform

4.1. Introduction

Zend_Cache fournit un moyen générique de mettre en cache des données.

Le cache dans le Zend Framework est réalisé via les frontends alors que les caches d'enregistrements sont stockés grâce à des adaptateurs de backend (File, Sqlite, Memcache...) grâce à un système souple d'IDs et de balises. En les utilisant, il est simple de supprimer des types spécifiques d'enregistrements par la suite (par exemple: "supprime tous les enregistrements de cache marqués avec une balise donnée").

Le coeur du module (Zend_Cache_Core) est générique, souple et configurable. Pour le moment, pour vos besoins spécifiques, il y a des frontends qui étendent Zend_Cache_Core pour votre confort : Output, File, Function et Class.

Exemple 4.1. Créer un frontend avec Zend_Cache::factory()

Zend_Cache::factory() instancie les objets corrects et les lie ensemble. Dans le premier exemple, nous allons utiliser le frontend Core avec le backend File

<?php
require_once 'Zend/Cache.php';

$frontendOptions = array(
   'lifetime' => 7200, // temps de vie du cache de 2 heures
   'automatic_serialization' => true
);

$backendOptions = array(
    'cache_dir' => './tmp/' // Répertoire où stocker les fichiers de cache
);

// créer un objet Zend_Cache_Core
$cache = Zend_Cache::factory('Core', 'File', $frontendOptions, $backendOptions);

?>

Maintenant que nous avons un frontend, nous pouvons mettre en cache tout type de données (nous avons activé la sérialisation). Par exemple nous pouvons mettre en cache le résultat d'une requête SQL couteuse. Après qu'il soit mis en cache, il n'y a plus besoin de se connecter à la base de données. Les enregistrements récupérés depuis le cache sont désérialisés.

<?php
// $cache initialisé dans l'exemple précédent

// on regarde si un cache existe déjà
if(!$result = $cache->get('myresult')) {

    // le cache est manquant, connexion à la base de données
    $db = Zend_Db::factory( [...] );

    $result = $db->fetchAll('SELECT * FROM grosse_table');

    $cache->save($result, 'myresult');

} else {

    // il y a un cache
    echo "Ceci est issu du cache !\n\n";

}

print_r($result);
?>

Exemple 4.2. Cache de sortie avec le frontend de sortie Zend_Cache

Nous marquons les sections dans lesquelles nous voulons un cache de sortie en ajoutant de la logique conditionnelle, en encapsulant la section entre les méthodes start() et end() (cela ressemble au premier exemple et est le coeur de la stratégie de mise en cache).

A l'intérieur, affichez vos données comme d'habitude toutes les sorties seront misent en cache quand la méthode end() est appelée. A la prochaine exécution, la section complète sera évitée, au profit de la récupération de son cache (si le cache est encore valide).

<?php
$frontendOptions = array(
   'lifetime' => 30,                  // temps de vue du cache de 30 secondes
   'automatic_serialization' => false  // par défaut
);

$backendOptions = array('cache_dir' => './tmp/');

$cache = Zend_Cache::factory('Output', 'File', $frontendOptions, $backendOptions);

// nous passons un identifiant unique de la méthode start()
if(!$cache->start('mypage')) {
    // affichage

    echo 'Hello world! ';
    echo 'Ceci est issu du cache('.time().') ';

    $cache->end(); // la sortie est sauvegardée est envoyé au navigateur
}

echo 'Ceci n\' jamais mis en cache ('.time().').';
?>       

Notez que nous affichons le résultat de time() deux fois ; c'est dans un but de démonstration. Essayez de lancer la page et de la rafraîchir plusieurs fois ; vous allez constater que le premier nombre ne change pas alors que le second change à chaque actualisation. C'est parce que le premier nombre a été mis en cache et sauvegardé. Après 30 secondes ('lifeTime' a été mis à 30 secondes), le premier nombre devrait de nouveau correspondre au second nombre parce que le cache a expiré -- seulement pour être mis en cache de nouveau. Vous devriez essayer ce code dans votre navigateur ou dans une console.

[Note] Note

Lorsque vous utilisez Zend_Cache, faîtes attention à l'identifiant du cache (passé à save() et start()). Il doit être unique pour chaque ressource que vous mettez en cache, sinon il est possible que des caches en efface d'autres, ou encore pire, s'affiche en lieu et place d'autres.