Rozdział 4. Zend_Cache

Spis treści

4.1. Wprowadzenie
4.2. Teoria buforowania
4.2.1. Metoda fabryki Zend_Cache
4.2.2. Nadawanie etykiet rekordom
4.2.3. Czyszczenie bufora
4.3. Frontendy 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. Backendy Zend_Cache
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. Wprowadzenie

Zend_Cache zapewnia ogólny sposób buforowania danych.

Buforowanie w Zend Framework jest przeprowadzane przez frontendy, a rekordy bufora są przechowywane za pomocą backendów (File, Sqlite, Memcache...) przy użyciu uniwersalnego systemu identyfikatorów ID oraz etykiet. Używając ich, łatwe jest kasowanie specyficznych typów rekordów (na przykład: "usuń wszystkie rekordy bufora oznaczone podaną etykietą").

Jądro modułu (Zend_Cache_Core) jest proste, uniwersalne i konfigurowalne. Obecnie, dla twoich specyficznych potrzeb dostępne są frontendy rozszerzające Zend_Cache_Core na przykład: Output, File, Function oraz Class.

Przykład 4.1. Pobieranie frontendu za pomocą Zend_Cache::factory()

Zend_Cache::factory() tworzy instancję odpowiedniego obiektu łączy je razem. W tym pierwszym przykładzie użyjemy frontendu Core wraz z backendem File.

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

$frontendOptions = array(
   'lifetime' => 7200, // okres ważności bufora 2 godziny
   'automatic_serialization' => true
);

$backendOptions = array(
    'cache_dir' => './tmp/' // Katalog w którym mają być składowane pliku bufora
);

// pobieranie obiektu Zend_Cache_Core
$cache = Zend_Cache::factory('Core', 'File', $frontendOptions, $backendOptions);

?>

Teraz gdy mamy frontend, możemy buforować dowolny typ danych (włączyliśmy serializację). Na przykład, możemy buforować rezultat bardzo obciążającego zapytania do bazy danych. Kiedy jest buforowane, nie ma nawet potrzeby aby łączyć się z bazą; rekordy są pobierane z bufora, a następnie odserializowane.

<?php

// obiekt $cache zainicjalizowany jak w poprzednim przykładzie

// sprawdzamy czy bufor istnieje:
if(!$result = $cache->load('myresult')) {

    // bufor nie istnieje; łączymy się z bazą

    $db = Zend_Db::factory( [...] );
    
    $result = $db->fetchAll('SELECT * FROM huge_table');
    
    $cache->save($result, 'myresult');
    
} else {

    // bufor istnieje! dajmy o tym znać
    echo "To pochodzi z bufora!\n\n";
    
}

print_r($result);

?>

Przykład 4.2. Buforowanie danych wyjściowych przy użyciu frontendu Zend_Cache

Sekcje w których chcemy buforować dane wyjściowe oznaczamy dodając pewną warunkową logikę, ograniczającą sekcję za pomocą metod start() oraz end() (to odpowiada pierwszemu przykładowi i jest główną strategią buforowania).

Wewnątrz wyświetlaj dane jak zawsze - wszystkie dane wyjściowe będą buforowane aż do napotkania metody end(). Podczas następnego wywołania, cała sekcja będzie ominięta, a użyte zostaną dane z bufora. (tak długo jak rekord bufora jest prawidłowy).

<?php

$frontendOptions = array(
   'lifetime' => 30,                  // okres ważności bufora pół minuty
   'automatic_serialization' => false  // to i tak jest domyślna wartość
);

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

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

// przekazujemy unikalny identyfikator do metody start()
if(!$cache->start('mypage')) {
    // wyświetlamy jak zawsze:
    
    echo 'Witajcie! ';
    echo 'To jest buforowane ('.time().') ';
    
    $cache->end(); // dane wyjściowe są zapisywane i wysyłane do przeglądarki
}

echo 'To nie jest nigdy buforowane ('.time().').';

?>       

Zauważ, że wyświetlamy rezultat funkcji time() dwa razy; jest to coś dynamicznego, aby zademenstrować przeznaczenie. Spróbuj uruchomić to i odświeżyć kilka razy; zauważysz, że pierwsza liczba się nie zmienia, a druga za każdym razem jest inna. Tak jest ponieważ pierwsza liczba była wyświetlona w sekcji buforowanej więc została zapisana. Po upływie pół minuty (ustawiliśmy okres ważności bufora na 30 sekund) powinny ponownie się zgadzać ponieważ bufor wygasł -- i został zapisany ponownie. Powinieneś to sprawdzić w swojej przeglądarce lub w konsoli.

[Notatka] Notatka

Kiedy używasz Zend_Cache, zwracaj uwagę na ważny identyfikator bufora (przekazany do metod save() oraz start()). Musi być unikalny dla każdego zasobum, który buforujesz, inaczej nie powiązane buforowane rekordy mogą się nawzajem ścierać, lub gorzej, jeden może wyświetlić się w miejscu drugiego.