Hibernation - Mise en cache

La mise en cache est un mécanisme permettant d'améliorer les performances d'un système. Il s'agit d'une mémoire tampon située entre l'application et la base de données. La mémoire cache stocke les éléments de données récemment utilisés afin de réduire autant que possible le nombre de hits dans la base de données.

La mise en cache est également importante pour Hibernate. Il utilise un schéma de mise en cache à plusieurs niveaux comme expliqué ci-dessous -

Cache de premier niveau

Le cache de premier niveau est le cache de session et est un cache obligatoire à travers lequel toutes les demandes doivent passer. L'objet Session conserve un objet sous son propre pouvoir avant de le valider dans la base de données.

Si vous émettez plusieurs mises à jour sur un objet, Hibernate essaie de retarder la mise à jour le plus longtemps possible pour réduire le nombre d'instructions SQL de mise à jour émises. Si vous fermez la session, tous les objets mis en cache sont perdus et conservés ou mis à jour dans la base de données.

Cache de deuxième niveau

Le cache de second niveau est un cache facultatif et le cache de premier niveau sera toujours consulté avant toute tentative de localisation d'un objet dans le cache de second niveau. Le cache de deuxième niveau peut être configuré par classe et par collection et principalement responsable de la mise en cache des objets entre les sessions.

Tout cache tiers peut être utilisé avec Hibernate. Unorg.hibernate.cache.CacheProvider interface est fournie, qui doit être implémentée pour fournir à Hibernate un descripteur de l'implémentation du cache.

Cache au niveau de la requête

Hibernate implémente également un cache pour les ensembles de résultats de requête qui s'intègre étroitement avec le cache de deuxième niveau.

Il s'agit d'une fonctionnalité facultative qui nécessite deux régions de cache physique supplémentaires qui contiennent les résultats de la requête en cache et les horodatages de la dernière mise à jour d'une table. Cela n'est utile que pour les requêtes exécutées fréquemment avec les mêmes paramètres.

Le cache de deuxième niveau

Hibernate utilise le cache de premier niveau par défaut et vous n'avez rien à faire pour utiliser le cache de premier niveau. Passons directement au cache optionnel de deuxième niveau. Toutes les classes ne bénéficient pas de la mise en cache, il est donc important de pouvoir désactiver le cache de deuxième niveau.

Le cache de deuxième niveau Hibernate est configuré en deux étapes. Tout d'abord, vous devez décider de la stratégie de concurrence à utiliser. Après cela, vous configurez l'expiration du cache et les attributs de cache physique à l'aide du fournisseur de cache.

Stratégies de concurrence

Une stratégie de concurrence est un médiateur, qui est chargé de stocker des éléments de données dans le cache et de les récupérer à partir du cache. Si vous envisagez d'activer un cache de second niveau, vous devrez décider, pour chaque classe et collection persistantes, de la stratégie de concurrence de cache à utiliser.

  • Transactional - Utilisez cette stratégie pour les données en lecture principale où il est essentiel d'éviter les données périmées dans les transactions simultanées, dans le cas rare d'une mise à jour.

  • Read-write - Utilisez à nouveau cette stratégie pour les données en lecture principale où il est essentiel d'éviter les données périmées dans les transactions simultanées, dans le cas rare d'une mise à jour.

  • Nonstrict-read-write- Cette stratégie ne garantit pas la cohérence entre le cache et la base de données. Utilisez cette stratégie si les données ne changent presque jamais et si une faible probabilité de données périmées n'est pas un problème critique.

  • Read-only- Une stratégie de concurrence adaptée aux données, qui ne change jamais. Utilisez-le uniquement pour les données de référence.

Si nous allons utiliser la mise en cache de second niveau pour notre Employee , ajoutons l'élément de mappage requis pour indiquer à Hibernate de mettre en cache les instances Employee en utilisant une stratégie de lecture-écriture.

<?xml version = "1.0" encoding = "utf-8"?>
<!DOCTYPE hibernate-mapping PUBLIC 
"-//Hibernate/Hibernate Mapping DTD//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd"> 
<hibernate-mapping>
   <class name = "Employee" table = "EMPLOYEE">
      
      <meta attribute = "class-description">
         This class contains the employee detail. 
      </meta>
      
      <cache usage = "read-write"/>
      
      <id name = "id" type = "int" column = "id">
         <generator class="native"/>
      </id>
      
      <property name = "firstName" column = "first_name" type = "string"/>
      <property name = "lastName" column = "last_name" type = "string"/>
      <property name = "salary" column = "salary" type = "int"/>
      
   </class>
</hibernate-mapping>

L'attribut usage = "lecture-écriture" indique à Hibernate d'utiliser une stratégie de concurrence en lecture-écriture pour le cache défini.

Fournisseur de cache

Votre prochaine étape après avoir examiné les stratégies de concurrence, vous utiliserez vos classes de candidats de cache pour choisir un fournisseur de cache. Hibernate vous oblige à choisir un seul fournisseur de cache pour l'ensemble de l'application.

N ° Sr. Nom et description du cache
1

EHCache

Il peut mettre en cache en mémoire ou sur disque et mise en cache en cluster et il prend en charge le cache de résultat de requête Hibernate en option.

2

OSCache

Prend en charge la mise en cache de la mémoire et du disque dans une seule machine virtuelle Java avec un ensemble complet de politiques d'expiration et la prise en charge du cache de requêtes.

3

warmCache

Un cache de cluster basé sur JGroups. Il utilise l'invalidation en cluster, mais ne prend pas en charge le cache de requêtes Hibernate.

4

JBoss Cache

Un cache en cluster répliqué entièrement transactionnel également basé sur la bibliothèque de multidiffusion JGroups. Il prend en charge la réplication ou l'invalidation, la communication synchrone ou asynchrone et le verrouillage optimiste et pessimiste. Le cache de requêtes Hibernate est pris en charge.

Chaque fournisseur de cache n'est pas compatible avec toutes les stratégies de concurrence. La matrice de compatibilité suivante vous aidera à choisir une combinaison appropriée.

Stratégie / Fournisseur Lecture seulement Nonstrictread-write Lire écrire Transactionnel
EHCache X X X  
OSCache X X X  
SwarmCache X X    
Cache JBoss X     X

Vous allez spécifier un fournisseur de cache dans le fichier de configuration hibernate.cfg.xml. Nous choisissons EHCache comme fournisseur de cache de second niveau -

<?xml version = "1.0" encoding = "utf-8"?>
<!DOCTYPE hibernate-configuration SYSTEM 
"http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
   <session-factory>
   
      <property name = "hibernate.dialect">
         org.hibernate.dialect.MySQLDialect
      </property>
   
      <property name = "hibernate.connection.driver_class">
         com.mysql.jdbc.Driver
      </property>
   
      <!-- Assume students is the database name -->
   
      <property name = "hibernate.connection.url">
         jdbc:mysql://localhost/test
      </property>
   
      <property name = "hibernate.connection.username">
         root
      </property>
   
      <property name = "hibernate.connection.password">
         root123
      </property>
   
      <property name = "hibernate.cache.provider_class">
         org.hibernate.cache.EhCacheProvider
      </property>
   
      <!-- List of XML mapping files -->
      <mapping resource = "Employee.hbm.xml"/>
   
   </session-factory>
</hibernate-configuration>

Maintenant, vous devez spécifier les propriétés des régions de cache. EHCache a son propre fichier de configuration,ehcache.xml, qui doit être dans le CLASSPATH de l'application. Une configuration de cache dans ehcache.xml pour la classe Employee peut ressembler à ceci -

<diskStore path="java.io.tmpdir"/>

<defaultCache
maxElementsInMemory = "1000"
eternal = "false"
timeToIdleSeconds = "120"
timeToLiveSeconds = "120"
overflowToDisk = "true"
/>

<cache name = "Employee"
maxElementsInMemory = "500"
eternal = "true"
timeToIdleSeconds = "0"
timeToLiveSeconds = "0"
overflowToDisk = "false"
/>

Voilà, maintenant nous avons activé la mise en cache de deuxième niveau pour la classe Employee et Hibernate, atteint désormais le cache de deuxième niveau chaque fois que vous accédez à un employé ou lorsque vous chargez un employé par identifiant.

Vous devez analyser toutes vos classes et choisir une stratégie de mise en cache appropriée pour chacune des classes. Parfois, la mise en cache de deuxième niveau peut dégrader les performances de l'application. Il est donc recommandé de comparer d'abord votre application, sans activer la mise en cache, puis d'activer votre mise en cache bien adaptée et de vérifier les performances. Si la mise en cache n'améliore pas les performances du système, il ne sert à rien d'activer tout type de mise en cache.

Le cache au niveau de la requête

Pour utiliser le cache de requêtes, vous devez d'abord l'activer à l'aide du hibernate.cache.use_query_cache="true"propriété dans le fichier de configuration. En définissant cette propriété sur true, vous obligez Hibernate à créer les caches nécessaires en mémoire pour contenir les ensembles de requêtes et d'identifiants.

Ensuite, pour utiliser le cache de requêtes, vous utilisez la méthode setCacheable (Boolean) de la classe Query. Par exemple -

Session session = SessionFactory.openSession();
Query query = session.createQuery("FROM EMPLOYEE");
query.setCacheable(true);
List users = query.list();
SessionFactory.closeSession();

Hibernate prend également en charge une prise en charge très fine du cache grâce au concept de région de cache. Une région de cache fait partie du cache qui a un nom.

Session session = SessionFactory.openSession();
Query query = session.createQuery("FROM EMPLOYEE");
query.setCacheable(true);
query.setCacheRegion("employee");
List users = query.list();
SessionFactory.closeSession();

Ce code utilise la méthode pour indiquer à Hibernate de stocker et de rechercher la requête dans la zone des employés du cache.