The Second Level Cache ====================== The Second Level Cache is designed to reduce the amount of necessary database access. It sits between your application and the database to avoid the number of database hits as many as possible. When turned on, entities will be first searched in cache and if they are not found, a database query will be fired an then the entity result will be stored in a cache provider. There are some flavors of caching available, but is better to cache read-only data. Be aware that caches are not aware of changes made to the persistent store by another application. They can, however, be configured to regularly expire cached data. Caching Regions --------------- Second level cache does not store instances of an entity, instead it caches only entity identifier and values. Each entity class, collection association and query has its region, where values of each instance are stored. Caching Regions are specific region into the cache provider that might store entities, collection or queries. Each cache region resides in a specific cache namespace and has its own lifetime configuration. Something like below for an entity region : .. code-block:: php ['id'=> 1, 'name' => 'FooBar', 'associationName'=>null], 'region_name:entity_2_hash' => ['id'=> 2, 'name' => 'Foo', 'associationName'=>['id'=>11]], 'region_name:entity_3_hash' => ['id'=> 3, 'name' => 'Bar', 'associationName'=>['id'=>22]] ]; If the entity holds a collection that also needs to be cached. An collection region could look something like : .. code-block:: php ['ownerId'=> 1, 'list' => [1, 2, 3]], 'region_name:entity_2_coll_assoc_name_hash' => ['ownerId'=> 2, 'list' => [2, 3]], 'region_name:entity_3_coll_assoc_name_hash' => ['ownerId'=> 3, 'list' => [2, 4]] ]; A query region might be something like : .. code-block:: php ['list' => [1, 2, 3]], 'region_name:query_2_hash' => ['list' => [2, 3]], 'region_name:query_3_hash' => ['list' => [2, 4]] ]; .. note:: Notice that when caching collection and queries only identifiers are stored. The entity values will be stored in its own region .. _reference-second-level-cache-regions: Cache Regions ------------- ``Doctrine\ORM\Cache\Region\DefaultRegion`` Its the default implementation. A simplest cache region compatible with all doctrine-cache drivers but does not support locking. ``Doctrine\ORM\Cache\Region`` and ``Doctrine\ORM\Cache\ConcurrentRegion`` Defines contracts that should be implemented by a cache provider. It allows you to provide your own cache implementation that might take advantage of specific cache driver. If you want to support locking for ``READ_WRITE`` strategies you should implement ``ConcurrentRegion``; ``CacheRegion`` otherwise. ``Doctrine\ORM\Cache\Region`` Defines a contract for accessing a particular cache region. .. code-block:: php setSecondLevelCacheEnabled(); //Cache factory $config->setSecondLevelCacheFactory($factory); Cache Factory ~~~~~~~~~~~~~ Cache Factory is the main point of extension. It allows you to provide a specific implementation of the following components : * ``QueryCache`` Store and retrieve query cache results. * ``CachedEntityPersister`` Store and retrieve entity results. * ``CachedCollectionPersister`` Store and retrieve query results. * ``EntityHydrator`` Transform an entity into a cache entry and cache entry into entities * ``CollectionHydrator`` Transform a collection into a cache entry and cache entry into collection .. code-block:: php setSecondLevelCacheRegionLifetime('my_entity_region', 3600); $config->setSecondLevelCacheDefaultRegionLifetime(7200); Cache Log ~~~~~~~~~ By providing a cache logger you should be able to get information about all cache operations such as hits, misses and puts. ``\Doctrine\ORM\Cache\Logging\StatisticsCacheLogger`` is a built-in implementation that provides basic statistics. .. code-block:: php setSecondLevelCacheLogger($logger); // Collect cache statistics // Get the number of entries successfully retrieved from a specific region. $logger->getRegionHitCount('my_entity_region'); // Get the number of cached entries *not* found in a specific region. $logger->getRegionMissCount('my_entity_region'); // Get the number of cacheable entries put in cache. $logger->getRegionPutCount('my_entity_region'); // Get the total number of put in all regions. $logger->getPutCount(); // Get the total number of entries successfully retrieved from all regions. $logger->getHitCount(); // Get the total number of cached entries *not* found in all regions. $logger->getMissCount(); If you want to get more information you should implement ``\Doctrine\ORM\Cache\Logging\CacheLogger``. and collect all information you want. .. code-block:: php .. code-block:: yaml Country: type: entity cache: usage : READ_ONLY region : my_entity_region id: id: type: integer id: true generator: strategy: IDENTITY fields: name: type: string Association cache definition ---------------------------- The most common use case is to cache entities. But we can also cache relationships. It caches the primary keys of association and cache each element will be cached into its region. .. configuration-block:: .. code-block:: php .. code-block:: yaml State: type: entity cache: usage : NONSTRICT_READ_WRITE id: id: type: integer id: true generator: strategy: IDENTITY fields: name: type: string manyToOne: state: targetEntity: Country joinColumns: country_id: referencedColumnName: id cache: usage : NONSTRICT_READ_WRITE oneToMany: cities: targetEntity:City mappedBy: state cache: usage : NONSTRICT_READ_WRITE Cache usage ~~~~~~~~~~~ Basic entity cache .. code-block:: php persist(new Country($name)); $em->flush(); // Hit database to insert the row and put into cache $em->clear(); // Clear entity manager $country = $em->find('Country', 1); // Retrieve item from cache $country->setName("New Name"); $em->persist($state); $em->flush(); // Hit database to update the row and update cache $em->clear(); // Clear entity manager $country = $em->find('Country', 1); // Retrieve item from cache Association cache .. code-block:: php persist(new State($name, $country)); $em->flush(); // Clear entity manager $em->clear(); // Retrieve item from cache $state = $em->find('State', 1); // Hit database to update the row and update cache entry $state->setName("New Name"); $em->persist($state); $em->flush(); // Create a new collection item $city = new City($name, $state); $state->addCity($city); // Hit database to insert new collection item, // put entity and collection cache into cache. $em->persist($city); $em->persist($state); $em->flush(); // Clear entity manager $em->clear(); // Retrieve item from cache $state = $em->find('State', 1); // Retrieve association from cache $country = $state->getCountry(); // Retrieve collection from cache $cities = $state->getCities(); echo $country->getName(); echo $state->getName(); // Retrieve each collection item from cache foreach ($cities as $city) { echo $city->getName(); } .. note:: Notice that all entities should be marked as cacheable. Using the query cache --------------------- The second level cache stores the entities, associations and collections. The query cache stores the results of the query but as identifiers, entity values are actually stored in the 2nd level cache. .. note:: Query cache should always be used in conjunction with the second-level-cache for those entities which should be cached. .. code-block:: php createQuery('SELECT c FROM Country c ORDER BY c.name') ->setCacheable(true) ->getResult(); // Check if query result is valid and load entities from cache $result2 = $em->createQuery('SELECT c FROM Country c ORDER BY c.name') ->setCacheable(true) ->getResult(); Cache API --------- Caches are not aware of changes made by another application. However, you can use the cache API to check / invalidate cache entries. .. code-block:: php getCache(); $cache->containsEntity('State', 1) // Check if the cache exists $cache->evictEntity('State', 1); // Remove an entity from cache $cache->evictEntityRegion('State'); // Remove all entities from cache $cache->containsCollection('State', 'cities', 1); // Check if the cache exists $cache->evictCollection('State', 'cities', 1); // Remove an entity collection from cache $cache->evictCollectionRegion('State', 'cities'); // Remove all collections from cache Limitations ----------- Composite primary key ~~~~~~~~~~~~~~~~~~~~~ .. note:: Composite primary key are supported by second level cache, however when one of the keys is an association the cached entity should always be retrieved using the association identifier. .. code-block:: php 1, 'target' => 2); $reference = $this->_em->find("Reference", $id); // NOT Supported $id = array('source' => new Article(1), 'target' => new Article(2)); $reference = $this->_em->find("Reference", $id); Concurrent cache region ~~~~~~~~~~~~~~~~~~~~~~~ A ``Doctrine\\ORM\\Cache\\ConcurrentRegion`` is designed to store concurrently managed data region. By default, Doctrine provides a very simple implementation based on file locks ``Doctrine\\ORM\\Cache\\Region\\FileLockRegion``. If you want to use an ``READ_WRITE`` cache, you should consider providing your own cache region. for more details about how to implement a cache region please see :ref:`reference-second-level-cache-regions`