Hibernate - Guide rapide

Qu'est-ce que JDBC?

JDBC signifie Java Database Connectivity. Il fournit un ensemble d'API Java pour accéder aux bases de données relationnelles à partir du programme Java. Ces API Java permettent aux programmes Java d'exécuter des instructions SQL et d'interagir avec n'importe quelle base de données compatible SQL.

JDBC fournit une architecture flexible pour écrire une application indépendante de la base de données qui peut s'exécuter sur différentes plates-formes et interagir avec différents SGBD sans aucune modification.

Avantages et inconvénients de JDBC

Avantages de JDBC Inconvénients de JDBC

Traitement SQL propre et simple

Bonnes performances avec des données volumineuses

Très bon pour les petites applications

Syntaxe simple si facile à apprendre

Complexe s'il est utilisé dans de grands projets

Grande surcharge de programmation

Pas d'encapsulation

Difficile à mettre en œuvre le concept MVC

La requête est spécifique au SGBD

Pourquoi Object Relational Mapping (ORM)?

Lorsque nous travaillons avec un système orienté objet, il y a un décalage entre le modèle objet et la base de données relationnelle. Les SGBDR représentent les données dans un format tabulaire tandis que les langages orientés objet, tels que Java ou C #, le représentent sous la forme d'un graphique d'objets interconnectés.

Considérez la classe Java suivante avec les constructeurs appropriés et la fonction publique associée -

public class Employee {
   private int id;
   private String first_name; 
   private String last_name;   
   private int salary;  

   public Employee() {}
   public Employee(String fname, String lname, int salary) {
      this.first_name = fname;
      this.last_name = lname;
      this.salary = salary;
   }
   
   public int getId() {
      return id;
   }
   
   public String getFirstName() {
      return first_name;
   }
   
   public String getLastName() {
      return last_name;
   }
   
   public int getSalary() {
      return salary;
   }
}

Considérez que les objets ci-dessus doivent être stockés et récupérés dans la table SGBDR suivante -

create table EMPLOYEE (
   id INT NOT NULL auto_increment,
   first_name VARCHAR(20) default NULL,
   last_name  VARCHAR(20) default NULL,
   salary     INT  default NULL,
   PRIMARY KEY (id)
);

Premier problème, que se passe-t-il si nous devons modifier le design de notre base de données après avoir développé quelques pages ou notre application? Deuxièmement, le chargement et le stockage d'objets dans une base de données relationnelle nous exposent aux cinq problèmes d'incompatibilité suivants:

Sr.No. Non-concordance et description
1

Granularity

Parfois, vous aurez un modèle objet, qui a plus de classes que le nombre de tables correspondantes dans la base de données.

2

Inheritance

Les SGBDR ne définissent rien de similaire à l'héritage, qui est un paradigme naturel dans les langages de programmation orientés objet.

3

Identity

Un SGBDR définit exactement une notion de «similitude»: la clé primaire. Java, cependant, définit à la fois l'identité d'objet (a == b) et l'égalité d'objet (a.equals (b)).

4

Associations

Les langages orientés objet représentent des associations utilisant des références d'objet tandis qu'un SGBDR représente une association en tant que colonne de clé étrangère.

5

Navigation

Les façons d'accéder aux objets en Java et dans le SGBDR sont fondamentalement différentes.

le Object-Rexaltation Mapping (ORM) est la solution pour gérer toutes les discordances d'impédance ci-dessus.

Qu'est-ce que l'ORM?

ORM signifie Object-Rexaltation Mapping (ORM) est une technique de programmation permettant de convertir des données entre des bases de données relationnelles et des langages de programmation orientés objet tels que Java, C #, etc.

Un système ORM présente les avantages suivants par rapport au JDBC ordinaire:

Sr.No. Avantages
1 Faisons accès aux objets de code métier plutôt qu'aux tables de base de données.
2 Masque les détails des requêtes SQL de la logique OO.
3 Basé sur JDBC «sous le capot».
4 Pas besoin de s'occuper de l'implémentation de la base de données.
5 Entités basées sur des concepts commerciaux plutôt que sur la structure de la base de données.
6 Gestion des transactions et génération automatique de clés.
sept Développement rapide de l'application.

Une solution ORM se compose des quatre entités suivantes -

Sr.No. Solutions
1 Une API pour effectuer des opérations CRUD de base sur des objets de classes persistantes.
2 Un langage ou une API pour spécifier les requêtes qui font référence aux classes et aux propriétés des classes.
3 Une fonction configurable pour spécifier les métadonnées de mappage.
4 Une technique pour interagir avec des objets transactionnels pour effectuer une vérification incorrecte, une récupération d'association paresseuse et d'autres fonctions d'optimisation.

Cadres Java ORM

Il existe plusieurs frameworks persistants et options ORM en Java. Un framework persistant est un service ORM qui stocke et récupère des objets dans une base de données relationnelle.

  • Beans d'entité Enterprise JavaBeans
  • Objets de données Java
  • Castor
  • TopLink
  • Printemps DAO
  • Hibernate
  • Et beaucoup plus

Hibernate est un Object-Rexaltation Msolution apping (ORM) pour JAVA. Il s'agit d'un framework persistant open source créé par Gavin King en 2001. Il s'agit d'un service de persistance et de requête relationnelle objet puissant et performant pour toute application Java.

Hibernate mappe les classes Java aux tables de base de données et des types de données Java aux types de données SQL et libère le développeur de 95% des tâches de programmation courantes liées à la persistance des données.

Hibernate se situe entre les objets Java traditionnels et le serveur de base de données pour gérer tous les travaux de persistance de ces objets en fonction des mécanismes et des modèles O / R appropriés.

Avantages Hibernate

  • Hibernate s'occupe de mapper les classes Java aux tables de base de données à l'aide de fichiers XML et sans écrire aucune ligne de code.

  • Fournit des API simples pour stocker et récupérer des objets Java directement vers et depuis la base de données.

  • En cas de modification dans la base de données ou dans une table, vous devez modifier uniquement les propriétés du fichier XML.

  • Abstractionne les types SQL inconnus et fournit un moyen de contourner les objets Java familiers.

  • Hibernate ne nécessite pas de serveur d'applications pour fonctionner.

  • Manipule les associations complexes d'objets de votre base de données.

  • Minimise l'accès à la base de données grâce à des stratégies de récupération intelligentes.

  • Fournit une interrogation simple des données.

Bases de données prises en charge

Hibernate prend en charge presque tous les principaux SGBDR. Voici une liste de quelques-uns des moteurs de base de données pris en charge par Hibernate -

  • Moteur de base de données HSQL
  • DB2/NT
  • MySQL
  • PostgreSQL
  • FrontBase
  • Oracle
  • Base de données Microsoft SQL Server
  • Sybase SQL Server
  • Serveur dynamique Informix

Technologies prises en charge

Hibernate prend en charge diverses autres technologies, notamment -

  • Ressort XDoclet
  • J2EE
  • Plug-ins Eclipse
  • Maven

Hibernate a une architecture en couches qui aide l'utilisateur à fonctionner sans avoir à connaître les API sous-jacentes. Hibernate utilise la base de données et les données de configuration pour fournir des services de persistance (et des objets persistants) à l'application.

Voici une vue de très haut niveau de l'architecture d'application Hibernate.

Voici une vue détaillée de l'architecture d'application Hibernate avec ses classes principales importantes.

Hibernate utilise diverses API Java existantes, telles que JDBC, Java Transaction API (JTA) et Java Naming and Directory Interface (JNDI). JDBC fournit un niveau rudimentaire d'abstraction de fonctionnalités communes aux bases de données relationnelles, permettant à presque toutes les bases de données avec un pilote JDBC d'être prises en charge par Hibernate. JNDI et JTA permettent à Hibernate d'être intégré aux serveurs d'applications J2EE.

La section suivante donne une brève description de chacun des objets de classe impliqués dans Hibernate Application Architecture.

Objet de configuration

L'objet Configuration est le premier objet Hibernate que vous créez dans une application Hibernate. Il n'est généralement créé qu'une seule fois lors de l'initialisation de l'application. Il représente un fichier de configuration ou de propriétés requis par Hibernate.

L'objet Configuration fournit deux composants clés -

  • Database Connection- Ceci est géré par un ou plusieurs fichiers de configuration pris en charge par Hibernate. Ces fichiers sonthibernate.properties et hibernate.cfg.xml.

  • Class Mapping Setup - Ce composant crée la connexion entre les classes Java et les tables de la base de données.

Objet SessionFactory

L'objet de configuration est utilisé pour créer un objet SessionFactory qui à son tour configure Hibernate pour l'application à l'aide du fichier de configuration fourni et permet d'instancier un objet Session. SessionFactory est un objet thread-safe et utilisé par tous les threads d'une application.

La SessionFactory est un objet lourd; il est généralement créé lors du démarrage de l'application et conservé pour une utilisation ultérieure. Vous auriez besoin d'un objet SessionFactory par base de données en utilisant un fichier de configuration distinct. Ainsi, si vous utilisez plusieurs bases de données, vous devrez créer plusieurs objets SessionFactory.

Objet de session

Une session est utilisée pour obtenir une connexion physique avec une base de données. L'objet Session est léger et conçu pour être instancié chaque fois qu'une interaction est nécessaire avec la base de données. Les objets persistants sont enregistrés et récupérés via un objet Session.

Les objets de session ne doivent pas rester ouverts pendant une longue période car ils ne sont généralement pas thread-safe et ils doivent être créés et détruits selon les besoins.

Objet de transaction

Une transaction représente une unité de travail avec la base de données et la plupart des SGBDR prennent en charge la fonctionnalité de transaction. Les transactions dans Hibernate sont gérées par un gestionnaire de transactions et une transaction sous-jacents (à partir de JDBC ou JTA).

Il s'agit d'un objet facultatif et les applications Hibernate peuvent choisir de ne pas utiliser cette interface, mais plutôt de gérer les transactions dans leur propre code d'application.

Objet de requête

Les objets de requête utilisent la chaîne SQL ou Hibernate Query Language (HQL) pour récupérer les données de la base de données et créer des objets. Une instance de requête est utilisée pour lier des paramètres de requête, limiter le nombre de résultats renvoyés par la requête et enfin exécuter la requête.

Objet Critères

Les objets Critères sont utilisés pour créer et exécuter des requêtes de critères orientés objet pour récupérer des objets.

Ce chapitre explique comment installer Hibernate et d'autres packages associés pour préparer un environnement pour les applications Hibernate. Nous travaillerons avec la base de données MySQL pour expérimenter des exemples Hibernate, alors assurez-vous que vous avez déjà une configuration pour la base de données MySQL. Pour plus de détails sur MySQL, vous pouvez consulter notre Tutoriel MySQL .

Téléchargement de Hibernate

Il est supposé que la dernière version de Java est déjà installée sur votre système. Voici les étapes simples pour télécharger et installer Hibernate sur votre système -

  • Choisissez si vous souhaitez installer Hibernate sous Windows ou Unix, puis passez à l'étape suivante pour télécharger le fichier .zip pour Windows et le fichier .tz pour Unix.

  • Téléchargez la dernière version d'Hibernate sur http://www.hibernate.org/downloads.

  • Au moment de la rédaction de ce tutoriel, j'ai téléchargé hibernate-distribution3.6.4.Final et lorsque vous décompressez le fichier téléchargé, il vous donnera la structure de répertoires comme indiqué dans l'image suivante

Installer Hibernate

Une fois que vous avez téléchargé et décompressé la dernière version du fichier d'installation Hibernate, vous devez suivre deux étapes simples. Assurez-vous que vous définissez correctement votre variable CLASSPATH, sinon vous rencontrerez des problèmes lors de la compilation de votre application.

  • Maintenant, copiez tous les fichiers de bibliothèque de /lib dans votre CLASSPATH et modifiez votre variable de chemin de classe pour inclure tous les JAR -

  • Enfin, copiez hibernate3.jarfichier dans votre CLASSPATH. Ce fichier se trouve dans le répertoire racine de l'installation et est le JAR principal dont Hibernate a besoin pour faire son travail.

Prérequis Hibernate

Voici la liste des packages / bibliothèques requis par Hibernate et vous devez les installer avant de commencer avec Hibernate. Pour installer ces packages, vous devrez copier les fichiers de bibliothèque depuis/lib dans votre CLASSPATH et modifiez votre variable CLASSPATH en conséquence.

Sr.No. Paquets / bibliothèques
1

dom4j

Analyse XML www.dom4j.org/

2

Xalan

Processeur XSLT https://xml.apache.org/xalan-j/

3

Xerces

L'analyseur Java Xerces https://xml.apache.org/xerces-j/

4

cglib

Modifications appropriées des classes Java lors de l'exécution http://cglib.sourceforge.net/

5

log4j

Journalisation Faremwork https://logging.apache.org/log4j

6

Commons

Journalisation, e-mail, etc. https://jakarta.apache.org/commons

sept

SLF4J

Façade de journalisation pour Java https://www.slf4j.org

Hibernate nécessite de savoir à l'avance - où trouver les informations de mappage qui définissent la relation entre vos classes Java et les tables de la base de données. Hibernate nécessite également un ensemble de paramètres de configuration liés à la base de données et à d'autres paramètres associés. Toutes ces informations sont généralement fournies sous forme de fichier de propriétés Java standard appeléhibernate.properties, ou sous forme de fichier XML nommé hibernate.cfg.xml.

Je considérerai un fichier au format XML hibernate.cfg.xmlpour spécifier les propriétés Hibernate requises dans mes exemples. La plupart des propriétés prennent leurs valeurs par défaut et il n'est pas nécessaire de les spécifier dans le fichier de propriétés sauf si cela est vraiment nécessaire. Ce fichier est conservé dans le répertoire racine du chemin de classe de votre application.

Propriétés Hibernate

Voici la liste des propriétés importantes, vous devrez configurer pour une base de données dans une situation autonome -

Sr.No. Propriétés et description
1

hibernate.dialect

Cette propriété permet à Hibernate de générer le SQL approprié pour la base de données choisie.

2

hibernate.connection.driver_class

La classe de pilote JDBC.

3

hibernate.connection.url

L'URL JDBC de l'instance de base de données.

4

hibernate.connection.username

Le nom d'utilisateur de la base de données.

5

hibernate.connection.password

Le mot de passe de la base de données.

6

hibernate.connection.pool_size

Limite le nombre de connexions en attente dans le pool de connexions à la base de données Hibernate.

sept

hibernate.connection.autocommit

Permet d'utiliser le mode autocommit pour la connexion JDBC.

Si vous utilisez une base de données avec un serveur d'applications et JNDI, vous devrez configurer les propriétés suivantes -

Sr.No. Propriétés et description
1

hibernate.connection.datasource

Le nom JNDI défini dans le contexte du serveur d'applications, que vous utilisez pour l'application.

2

hibernate.jndi.class

La classe InitialContext pour JNDI.

3

hibernate.jndi.<JNDIpropertyname>

Passe toute propriété JNDI de votre choix au JNDI InitialContext .

4

hibernate.jndi.url

Fournit l'URL de JNDI.

5

hibernate.connection.username

Le nom d'utilisateur de la base de données.

6

hibernate.connection.password

Le mot de passe de la base de données.

Mettre en veille prolongée avec la base de données MySQL

MySQL est l'un des systèmes de base de données open source les plus populaires actuellement disponibles. Laissez-nous créerhibernate.cfg.xmlfichier de configuration et placez-le à la racine du chemin de classe de votre application. Vous devrez vous assurer que vous aveztestdb base de données disponible dans votre base de données MySQL et vous avez un utilisateur test disponible pour accéder à la base de données.

Le fichier de configuration XML doit être conforme à la DTD de configuration Hibernate 3, disponible sur http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd.

<?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 test 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>
      
      <!-- List of XML mapping files -->
      <mapping resource = "Employee.hbm.xml"/>
      
   </session-factory>
</hibernate-configuration>

Le fichier de configuration ci-dessus comprend <mapping> les balises, qui sont liées au fichier de mise en veille prolongée et nous verrons dans le chapitre suivant ce qu'est exactement un fichier de mise en veille prolongée et comment et pourquoi l'utilisons-nous?

Voici la liste des différents types de propriété de dialecte de bases de données importantes -

Sr.No. Propriété de base de données et de dialecte
1

DB2

org.hibernate.dialect.DB2Dialect

2

HSQLDB

org.hibernate.dialect.HSQLDialect

3

HypersonicSQL

org.hibernate.dialect.HSQLDialect

4

Informix

org.hibernate.dialect.InformixDialect

5

Ingres

org.hibernate.dialect.IngresDialect

6

Interbase

org.hibernate.dialect.InterbaseDialect

sept

Microsoft SQL Server 2000

org.hibernate.dialect.SQLServerDialect

8

Microsoft SQL Server 2005

org.hibernate.dialect.SQLServer2005Dialect

9

Microsoft SQL Server 2008

org.hibernate.dialect.SQLServer2008Dialect

dix

MySQL

org.hibernate.dialect.MySQLDialect

11

Oracle (any version)

org.hibernate.dialect.OracleDialect

12

Oracle 11g

org.hibernate.dialect.Oracle10gDialect

13

Oracle 10g

org.hibernate.dialect.Oracle10gDialect

14

Oracle 9i

org.hibernate.dialect.Oracle9iDialect

15

PostgreSQL

org.hibernate.dialect.PostgreSQLDialect

16

Progress

org.hibernate.dialect.ProgressDialect

17

SAP DB

org.hibernate.dialect.SAPDBDialect

18

Sybase

org.hibernate.dialect.SybaseDialect

19

Sybase Anywhere

org.hibernate.dialect.SybaseAnywhereDialect

Une session est utilisée pour obtenir une connexion physique avec une base de données. L'objet Session est léger et conçu pour être instancié chaque fois qu'une interaction est nécessaire avec la base de données. Les objets persistants sont enregistrés et récupérés via un objet Session.

Les objets de session ne doivent pas rester ouverts pendant une longue période car ils ne sont généralement pas thread-safe et ils doivent être créés et détruits selon les besoins. La fonction principale de la session est d'offrir, de créer, de lire et de supprimer des opérations pour les instances de classes d'entités mappées.

Les instances peuvent exister dans l'un des trois états suivants à un moment donné -

  • transient - Une nouvelle instance d'une classe persistante, qui n'est pas associée à une session et n'a pas de représentation dans la base de données et aucune valeur d'identifiant est considérée comme transitoire par Hibernate.

  • persistent- Vous pouvez rendre une instance transitoire persistante en l'associant à une session. Une instance persistante a une représentation dans la base de données, une valeur d'identifiant et est associée à une session.

  • detached - Une fois que nous fermons la session Hibernate, l'instance persistante deviendra une instance détachée.

Une instance de Session est sérialisable si ses classes persistantes sont sérialisables. Une transaction typique doit utiliser l'idiome suivant -

Session session = factory.openSession();
Transaction tx = null;

try {
   tx = session.beginTransaction();
   // do some work
   ...
   tx.commit();
}

catch (Exception e) {
   if (tx!=null) tx.rollback();
   e.printStackTrace(); 
} finally {
   session.close();
}

Si la session lève une exception, la transaction doit être annulée et la session doit être supprimée.

Méthodes d'interface de session

Il existe un certain nombre de méthodes fournies par le Sessioninterface, mais je vais énumérer uniquement quelques méthodes importantes, que nous utiliserons dans ce didacticiel. Vous pouvez consulter la documentation Hibernate pour une liste complète des méthodes associées àSession et SessionFactory.

Sr.No. Méthodes de session et description
1

Transaction beginTransaction()

Commencez une unité de travail et renvoyez l'objet Transaction associé.

2

void cancelQuery()

Annuler l'exécution de la requête en cours.

3

void clear()

Effacez complètement la session.

4

Connection close()

Terminez la session en libérant la connexion JDBC et en nettoyant.

5

Criteria createCriteria(Class persistentClass)

Créez une nouvelle instance Criteria, pour la classe d'entité donnée, ou une superclasse d'une classe d'entité.

6

Criteria createCriteria(String entityName)

Créez une nouvelle instance Criteria, pour le nom d'entité donné.

sept

Serializable getIdentifier(Object object)

Renvoie la valeur d'identifiant de l'entité donnée associée à cette session.

8

Query createFilter(Object collection, String queryString)

Créez une nouvelle instance de Query pour la collection et la chaîne de filtre données.

9

Query createQuery(String queryString)

Créez une nouvelle instance de Query pour la chaîne de requête HQL donnée.

dix

SQLQuery createSQLQuery(String queryString)

Créez une nouvelle instance de SQLQuery pour la chaîne de requête SQL donnée.

11

void delete(Object object)

Supprimez une instance persistante de la banque de données.

12

void delete(String entityName, Object object)

Supprimez une instance persistante de la banque de données.

13

Session get(String entityName, Serializable id)

Renvoie l'instance persistante de l'entité nommée donnée avec l'identifiant donné, ou null s'il n'y a pas d'instance persistante de ce type.

14

SessionFactory getSessionFactory()

Obtenez la fabrique de sessions qui a créé cette session.

15

void refresh(Object object)

Relisez l'état de l'instance donnée à partir de la base de données sous-jacente.

16

Transaction getTransaction()

Obtenez l'instance de transaction associée à cette session.

17

boolean isConnected()

Vérifiez si la session est actuellement connectée.

18

boolean isDirty()

Cette session contient-elle des modifications qui doivent être synchronisées avec la base de données?

19

boolean isOpen()

Vérifiez si la session est toujours ouverte.

20

Serializable save(Object object)

Persiste l'instance transitoire donnée, en affectant d'abord un identifiant généré.

21

void saveOrUpdate(Object object)

Sauvegardez (Object) ou mettez à jour (Object) l'instance donnée.

22

void update(Object object)

Mettez à jour l'instance persistante avec l'identifiant de l'instance détachée donnée.

23

void update(String entityName, Object object)

Mettez à jour l'instance persistante avec l'identifiant de l'instance détachée donnée.

Le concept entier d'Hibernate est de prendre les valeurs des attributs de classe Java et de les conserver dans une table de base de données. Un document de mappage aide Hibernate à déterminer comment extraire les valeurs des classes et les mapper avec la table et les champs associés.

Les classes Java dont les objets ou les instances seront stockés dans des tables de base de données sont appelées classes persistantes dans Hibernate. Hibernate fonctionne mieux si ces classes suivent des règles simples, également connues sous le nom dePlain Old Java Object (POJO) modèle de programmation.

Les règles principales des classes persistantes sont les suivantes, cependant, aucune de ces règles n'est une exigence stricte -

  • Toutes les classes Java qui seront persistantes ont besoin d'un constructeur par défaut.

  • Toutes les classes doivent contenir un identifiant afin de permettre une identification facile de vos objets dans Hibernate et la base de données. Cette propriété correspond à la colonne de clé primaire d'une table de base de données.

  • Tous les attributs qui seront conservés doivent être déclarés privés et avoir getXXX et setXXX méthodes définies dans le style JavaBean.

  • Une fonctionnalité centrale d'Hibernate, les proxies, dépend du fait que la classe persistante est soit non finale, soit de l'implémentation d'une interface qui déclare toutes les méthodes publiques.

  • Toutes les classes qui n'étendent pas ou n'implémentent pas certaines classes et interfaces spécialisées requises par le framework EJB.

Le nom POJO est utilisé pour souligner qu'un objet donné est un objet Java ordinaire, pas un objet spécial, et en particulier pas un Enterprise JavaBean.

Exemple POJO simple

Sur la base des quelques règles mentionnées ci-dessus, nous pouvons définir une classe POJO comme suit -

public class Employee {
   private int id;
   private String firstName; 
   private String lastName;   
   private int salary;  

   public Employee() {}
   public Employee(String fname, String lname, int salary) {
      this.firstName = fname;
      this.lastName = lname;
      this.salary = salary;
   }
   
   public int getId() {
      return id;
   }
   
   public void setId( int id ) {
      this.id = id;
   }
   
   public String getFirstName() {
      return firstName;
   }
   
   public void setFirstName( String first_name ) {
      this.firstName = first_name;
   }
   
   public String getLastName() {
      return lastName;
   }
   
   public void setLastName( String last_name ) {
      this.lastName = last_name;
   }
   
   public int getSalary() {
      return salary;
   }
   
   public void setSalary( int salary ) {
      this.salary = salary;
   }
}

Les mappages objet / relationnel sont généralement définis dans un document XML. Ce fichier de mappage indique à Hibernate - comment mapper la ou les classes définies aux tables de la base de données?

Bien que de nombreux utilisateurs d'Hibernate choisissent d'écrire le XML à la main, un certain nombre d'outils existent pour générer le document de mappage. Ceux-ci inclusXDoclet, Middlegen et AndroMDA pour les utilisateurs avancés d'Hibernate.

Considérons notre classe POJO précédemment définie dont les objets persisteront dans la table définie dans la section suivante.

public class Employee {
   private int id;
   private String firstName; 
   private String lastName;   
   private int salary;  

   public Employee() {}
   
   public Employee(String fname, String lname, int salary) {
      this.firstName = fname;
      this.lastName = lname;
      this.salary = salary;
   }
   
   public int getId() {
      return id;
   }
   
   public void setId( int id ) {
      this.id = id;
   }
   
   public String getFirstName() {
      return firstName;
   }
   
   public void setFirstName( String first_name ) {
      this.firstName = first_name;
   }
   
   public String getLastName() {
      return lastName;
   }
   
   public void setLastName( String last_name ) {
      this.lastName = last_name;
   }
   
   public int getSalary() {
      return salary;
   }
   
   public void setSalary( int salary ) {
      this.salary = salary;
   }
}

Il y aurait une table correspondant à chaque objet que vous êtes prêt à fournir de la persistance. Considérez que les objets ci-dessus doivent être stockés et récupérés dans la table SGBDR suivante -

create table EMPLOYEE (
   id INT NOT NULL auto_increment,
   first_name VARCHAR(20) default NULL,
   last_name  VARCHAR(20) default NULL,
   salary     INT  default NULL,
   PRIMARY KEY (id)
);

Sur la base des deux entités ci-dessus, nous pouvons définir le fichier de mappage suivant, qui indique à Hibernate comment mapper la ou les classes définies aux tables de la base de données.

<?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>
      
      <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>

Vous devez enregistrer le document de mappage dans un fichier au format <classname> .hbm.xml. Nous avons enregistré notre document de cartographie dans le fichier Employee.hbm.xml.

Voyons comprendre un petit détail sur les éléments de mappage utilisés dans le fichier de mappage -

  • Le document de mappage est un document XML ayant <hibernate-mapping> comme élément racine, qui contient tous les <class> éléments.

  • le <class>Les éléments sont utilisés pour définir des mappages spécifiques d'une classe Java vers les tables de la base de données. Le nom de la classe Java est spécifié à l'aide duname attribut de l'élément de classe et de la base de données table nom est spécifié à l'aide de l'attribut table.

  • le <meta> element est un élément facultatif et peut être utilisé pour créer la description de classe.

  • le <id>L'élément mappe l'attribut ID unique de la classe à la clé primaire de la table de base de données. lename l'attribut de l'élément id fait référence à la propriété de la classe et le columnL'attribut fait référence à la colonne de la table de base de données. letype L'attribut contient le type de mappage hibernate, ces types de mappage convertiront du type de données Java en type de données SQL.

  • le <generator>L'élément dans l'élément id est utilisé pour générer automatiquement les valeurs de clé primaire. leclass l'attribut de l'élément générateur est défini sur native pour laisser hiberner reprendre soit identity, sequence, ou hilo algorithme pour créer une clé primaire en fonction des capacités de la base de données sous-jacente.

  • le <property>L'élément est utilisé pour mapper une propriété de classe Java à une colonne de la table de base de données. lename l'attribut de l'élément fait référence à la propriété de la classe et le columnL'attribut fait référence à la colonne de la table de base de données. letype L'attribut contient le type de mappage hibernate, ces types de mappage convertiront du type de données Java en type de données SQL.

Il existe d'autres attributs et éléments disponibles, qui seront utilisés dans un document de cartographie et j'essaierais d'en couvrir autant que possible tout en discutant d'autres sujets liés à Hibernate.

Lorsque vous préparez un document de mappage Hibernate, vous constatez que vous mappez les types de données Java en types de données SGBDR. letypesdéclarés et utilisés dans les fichiers de mappage ne sont pas des types de données Java; ce ne sont pas non plus des types de base de données SQL. Ces types sont appelésHibernate mapping types, qui peut traduire des types de données Java en SQL et vice versa.

Ce chapitre répertorie tous les types de mappage de base, date et heure, grands objets et divers autres types de mappage intégrés.

Types primitifs

Type de mappage Type Java Type SQL ANSI
entier int ou java.lang.Integer ENTIER
longue long ou java.lang.Long GRAND
court short ou java.lang.Short PETITE MENTHE
flotte float ou java.lang.Float FLOTTE
double double ou java.lang.Double DOUBLE
big_decimal java.math.BigDecimal NUMÉRIQUE
personnage java.lang.String CHAR (1)
chaîne java.lang.String VARCHAR
octet byte ou java.lang.Byte TINYINT
booléen booléen ou java.lang.Boolean BIT
Oui Non booléen ou java.lang.Boolean CHAR (1) ('Y' ou 'N')
vrai faux booléen ou java.lang.Boolean CHAR (1) ('T' ou 'F')

Types de date et d'heure

Type de mappage Type Java Type SQL ANSI
Date java.util.Date ou java.sql.Date DATE
temps java.util.Date ou java.sql.Time TEMPS
horodatage java.util.Date ou java.sql.Timestamp HORAIRE
calendrier java.util.Calendar HORAIRE
calendrier_date java.util.Calendar DATE

Types d'objets binaires et grands

Type de mappage Type Java Type SQL ANSI
binaire octet[] VARBINAIRE (ou BLOB)
texte java.lang.String CLOB
sérialisable toute classe Java qui implémente java.io.Serializable VARBINAIRE (ou BLOB)
clob java.sql.Clob CLOB
goutte java.sql.Blob GOUTTE

Types liés au JDK

Type de mappage Type Java Type SQL ANSI
classe java.lang.Class VARCHAR
lieu java.util.Locale VARCHAR
fuseau horaire java.util.TimeZone VARCHAR
devise java.util.Currency VARCHAR

Prenons maintenant un exemple pour comprendre comment nous pouvons utiliser Hibernate pour fournir la persistance Java dans une application autonome. Nous passerons par les différentes étapes impliquées dans la création d'une application Java utilisant la technologie Hibernate.

Créer des classes POJO

La première étape de la création d'une application consiste à créer la ou les classes Java POJO, en fonction de l'application qui sera conservée dans la base de données. Considérons notreEmployee classe avec getXXX et setXXX méthodes pour en faire une classe compatible JavaBeans.

Un POJO (Plain Old Java Object) est un objet Java qui n'étend ni n'implémente certaines classes et interfaces spécialisées respectivement requises par le framework EJB. Tous les objets Java normaux sont POJO.

Lorsque vous concevez une classe à persister par Hibernate, il est important de fournir un code compatible JavaBeans ainsi qu'un attribut, qui fonctionnerait comme un index comme id attribut dans la classe Employee.

public class Employee {
   private int id;
   private String firstName; 
   private String lastName;   
   private int salary;  

   public Employee() {}
   public Employee(String fname, String lname, int salary) {
      this.firstName = fname;
      this.lastName = lname;
      this.salary = salary;
   }
   
   public int getId() {
      return id;
   }
   
   public void setId( int id ) {
      this.id = id;
   }
   
   public String getFirstName() {
      return firstName;
   }
   
   public void setFirstName( String first_name ) {
      this.firstName = first_name;
   }
   
   public String getLastName() {
      return lastName;
   }
   
   public void setLastName( String last_name ) {
      this.lastName = last_name;
   }
   
   public int getSalary() {
      return salary;
   }
   
   public void setSalary( int salary ) {
      this.salary = salary;
   }
}

Créer des tables de base de données

La deuxième étape serait de créer des tables dans votre base de données. Il y aurait une table correspondant à chaque objet, vous êtes prêt à fournir la persistance. Considérez que les objets ci-dessus doivent être stockés et récupérés dans la table SGBDR suivante -

create table EMPLOYEE (
   id INT NOT NULL auto_increment,
   first_name VARCHAR(20) default NULL,
   last_name  VARCHAR(20) default NULL,
   salary     INT  default NULL,
   PRIMARY KEY (id)
);

Créer un fichier de configuration de mappage

Cette étape consiste à créer un fichier de mappage qui indique à Hibernate comment mapper la ou les classes définies aux tables de la base de données.

<?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>
      
      <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>

Vous devez enregistrer le document de mappage dans un fichier au format <classname> .hbm.xml. Nous avons enregistré notre document de cartographie dans le fichier Employee.hbm.xml. Voyons quelques détails sur le document cartographique -

  • Le document de mappage est un document XML ayant <hibernate-mapping> comme élément racine qui contient tous les éléments <class>.

  • le <class>Les éléments sont utilisés pour définir des mappages spécifiques d'une classe Java vers les tables de la base de données. Le nom de la classe Java est spécifié à l'aide duname l'attribut de l'élément de classe et le nom de la table de base de données sont spécifiés à l'aide de table attribut.

  • le <meta> element est un élément facultatif et peut être utilisé pour créer la description de classe.

  • le <id>L'élément mappe l'attribut ID unique de la classe à la clé primaire de la table de base de données. lename l'attribut de l'élément id fait référence à la propriété de la classe et le columnL'attribut fait référence à la colonne de la table de base de données. letype L'attribut contient le type de mappage hibernate, ces types de mappage convertiront du type de données Java en type de données SQL.

  • le <generator>L'élément dans l'élément id est utilisé pour générer automatiquement les valeurs de clé primaire. leclass l'attribut de l'élément générateur est défini sur native pour laisser hiberner reprendre soit identity, sequence ou hilo algorithme pour créer une clé primaire en fonction des capacités de la base de données sous-jacente.

  • le <property>L'élément est utilisé pour mapper une propriété de classe Java à une colonne de la table de base de données. lename l'attribut de l'élément fait référence à la propriété de la classe et le columnL'attribut fait référence à la colonne de la table de base de données. letype L'attribut contient le type de mappage hibernate, ces types de mappage convertiront du type de données Java en type de données SQL.

Il existe d'autres attributs et éléments disponibles, qui seront utilisés dans un document de cartographie et j'essaierais d'en couvrir autant que possible tout en discutant d'autres sujets liés à Hibernate.

Créer une classe d'application

Enfin, nous allons créer notre classe d'application avec la méthode main () pour exécuter l'application. Nous utiliserons cette application pour sauvegarder quelques enregistrements d'employés, puis nous appliquerons les opérations CRUD sur ces enregistrements.

import java.util.List; 
import java.util.Date;
import java.util.Iterator; 
 
import org.hibernate.HibernateException; 
import org.hibernate.Session; 
import org.hibernate.Transaction;
import org.hibernate.SessionFactory;
import org.hibernate.cfg.Configuration;

public class ManageEmployee {
   private static SessionFactory factory; 
   public static void main(String[] args) {
      
      try {
         factory = new Configuration().configure().buildSessionFactory();
      } catch (Throwable ex) { 
         System.err.println("Failed to create sessionFactory object." + ex);
         throw new ExceptionInInitializerError(ex); 
      }
      
      ManageEmployee ME = new ManageEmployee();

      /* Add few employee records in database */
      Integer empID1 = ME.addEmployee("Zara", "Ali", 1000);
      Integer empID2 = ME.addEmployee("Daisy", "Das", 5000);
      Integer empID3 = ME.addEmployee("John", "Paul", 10000);

      /* List down all the employees */
      ME.listEmployees();

      /* Update employee's records */
      ME.updateEmployee(empID1, 5000);

      /* Delete an employee from the database */
      ME.deleteEmployee(empID2);

      /* List down new list of the employees */
      ME.listEmployees();
   }
   
   /* Method to CREATE an employee in the database */
   public Integer addEmployee(String fname, String lname, int salary){
      Session session = factory.openSession();
      Transaction tx = null;
      Integer employeeID = null;
      
      try {
         tx = session.beginTransaction();
         Employee employee = new Employee(fname, lname, salary);
         employeeID = (Integer) session.save(employee); 
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
      return employeeID;
   }
   
   /* Method to  READ all the employees */
   public void listEmployees( ){
      Session session = factory.openSession();
      Transaction tx = null;
      
      try {
         tx = session.beginTransaction();
         List employees = session.createQuery("FROM Employee").list(); 
         for (Iterator iterator = employees.iterator(); iterator.hasNext();){
            Employee employee = (Employee) iterator.next(); 
            System.out.print("First Name: " + employee.getFirstName()); 
            System.out.print("  Last Name: " + employee.getLastName()); 
            System.out.println("  Salary: " + employee.getSalary()); 
         }
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
   }
   
   /* Method to UPDATE salary for an employee */
   public void updateEmployee(Integer EmployeeID, int salary ){
      Session session = factory.openSession();
      Transaction tx = null;
      
      try {
         tx = session.beginTransaction();
         Employee employee = (Employee)session.get(Employee.class, EmployeeID); 
         employee.setSalary( salary );
		 session.update(employee); 
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
   }
   
   /* Method to DELETE an employee from the records */
   public void deleteEmployee(Integer EmployeeID){
      Session session = factory.openSession();
      Transaction tx = null;
      
      try {
         tx = session.beginTransaction();
         Employee employee = (Employee)session.get(Employee.class, EmployeeID); 
         session.delete(employee); 
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
   }
}

Compilation et exécution

Voici les étapes pour compiler et exécuter l'application mentionnée ci-dessus. Assurez-vous que vous avez correctement défini PATH et CLASSPATH avant de procéder à la compilation et à l'exécution.

  • Créez le fichier de configuration hibernate.cfg.xml comme expliqué dans le chapitre de configuration.

  • Créez le fichier de mappage Employee.hbm.xml comme indiqué ci-dessus.

  • Créez le fichier source Employee.java comme indiqué ci-dessus et compilez-le.

  • Créez le fichier source ManageEmployee.java comme indiqué ci-dessus et compilez-le.

  • Exécutez le binaire ManageEmployee pour exécuter le programme.

Vous obtiendrez le résultat suivant et les enregistrements seraient créés dans la table EMPLOYEE.

$java ManageEmployee
.......VARIOUS LOG MESSAGES WILL DISPLAY HERE........

First Name: Zara  Last Name: Ali  Salary: 1000
First Name: Daisy  Last Name: Das  Salary: 5000
First Name: John  Last Name: Paul  Salary: 10000
First Name: Zara  Last Name: Ali  Salary: 5000
First Name: John  Last Name: Paul  Salary: 10000

Si vous vérifiez votre table EMPLOYEE, elle doit contenir les enregistrements suivants -

mysql> select * from EMPLOYEE;
+----+------------+-----------+--------+
| id | first_name | last_name | salary |
+----+------------+-----------+--------+
| 29 | Zara       | Ali       |   5000 |
| 31 | John       | Paul      |  10000 |
+----+------------+-----------+--------+
2 rows in set (0.00 sec

mysql>

Jusqu'à présent, nous avons vu un mappage O / R très basique utilisant la mise en veille prolongée, mais il y a trois sujets de mappage les plus importants, que nous devons apprendre en détail.

Ce sont -

  • Cartographie des collections,
  • Mappage des associations entre les classes d'entités, et
  • Mappages de composants.

Mappages de collections

Si une entité ou une classe a une collection de valeurs pour une variable particulière, nous pouvons mapper ces valeurs à l'aide de l'une des interfaces de collection disponibles dans java. Hibernate peut conserver des instances dejava.util.Map, java.util.Set, java.util.SortedMap, java.util.SortedSet, java.util.List, et n'importe quel array d'entités ou de valeurs persistantes.

Sr.No. Type de collection et description du mappage
1 java.util.Set

Ceci est mappé avec un élément <set> et initialisé avec java.util.HashSet

2 java.util.SortedSet

Ceci est mappé avec un élément <set> et initialisé avec java.util.TreeSet. lesort L'attribut peut être défini sur un comparateur ou un ordre naturel.

3 java.util.List

Ceci est mappé avec un élément <list> et initialisé avec java.util.ArrayList

4 java.util.Collection

Ceci est mappé avec un élément <bag> ou <ibag> et initialisé avec java.util.ArrayList

5 java.util.Map

Ceci est mappé avec un élément <map> et initialisé avec java.util.HashMap

6 java.util.SortedMap

Ceci est mappé avec un élément <map> et initialisé avec java.util.TreeMap. lesort L'attribut peut être défini sur un comparateur ou un ordre naturel.

Les tableaux sont pris en charge par Hibernate avec <primitive-array> pour les types de valeurs primitives Java et <array> pour tout le reste. Cependant, ils sont rarement utilisés, je ne vais donc pas en parler dans ce tutoriel.

Si vous souhaitez mapper une interface de collection définie par l'utilisateur, qui n'est pas directement prise en charge par Hibernate, vous devez informer Hibernate de la sémantique de vos collections personnalisées, ce qui n'est pas très facile et ne recommande pas l'utilisation.

Mappages d'association

Le mappage des associations entre les classes d'entités et les relations entre les tables est l'âme d'ORM. Voici les quatre façons dont la cardinalité de la relation entre les objets peut être exprimée. Un mappage d'association peut être unidirectionnel ou bidirectionnel.

Sr.No. Type de mappage et description
1 Plusieurs à un

Cartographie de la relation plusieurs à un à l'aide d'Hibernate

2 Un par un

Cartographie des relations individuelles à l'aide d'Hibernate

3 Un-à-plusieurs

Cartographie de la relation un-à-plusieurs avec Hibernate

4 Plusieurs à plusieurs

Cartographie de la relation plusieurs à plusieurs à l'aide d'Hibernate

Mappages de composants

Il est très possible qu'une classe Entity puisse avoir une référence à une autre classe en tant que variable membre. Si la classe référencée n'a pas son propre cycle de vie et dépend complètement du cycle de vie de la classe d'entité propriétaire, alors la classe référencée est donc appeléeComponent class.

Le mappage de la collection de composants est également possible de la même manière que le mappage de collections régulières avec des différences de configuration mineures. Nous verrons ces deux mappages en détail avec des exemples.

Sr.No. Type de mappage et description
1 Mappages de composants

Mappage pour une classe ayant une référence à une autre classe en tant que variable membre.

Jusqu'à présent, vous avez vu comment Hibernate utilise un fichier de mappage XML pour la transformation des données de POJO en tables de base de données et vice versa. Les annotations Hibernate sont le moyen le plus récent de définir des mappages sans utiliser de fichier XML. Vous pouvez utiliser des annotations en plus ou en remplacement des métadonnées de mappage XML.

Les annotations Hibernate sont le moyen puissant de fournir les métadonnées pour le mappage d'objets et de tables relationnelles. Toutes les métadonnées sont écrites dans le fichier java POJO avec le code, cela aide l'utilisateur à comprendre la structure de la table et POJO simultanément pendant le développement.

Si vous souhaitez rendre votre application portable vers d'autres applications ORM compatibles EJB 3, vous devez utiliser des annotations pour représenter les informations de mappage, mais si vous voulez une plus grande flexibilité, vous devriez opter pour des mappages basés sur XML.

Configuration de l'environnement pour l'annotation Hibernate

Tout d'abord, vous devez vous assurer que vous utilisez JDK 5.0, sinon vous devez mettre à niveau votre JDK vers JDK 5.0 pour profiter de la prise en charge native des annotations.

Deuxièmement, vous devrez installer le package de distribution des annotations Hibernate 3.x, disponible sur le sourceforge: ( Télécharger l'annotation Hibernate ) et copierhibernate-annotations.jar, lib/hibernate-comons-annotations.jar et lib/ejb3-persistence.jar de la distribution Hibernate Annotations à votre CLASSPATH.

Exemple de classe annotée

Comme je l'ai mentionné ci-dessus tout en travaillant avec Hibernate Annotation, toutes les métadonnées sont regroupées dans le fichier java POJO avec le code, cela aide l'utilisateur à comprendre la structure de la table et POJO simultanément pendant le développement.

Considérez que nous allons utiliser la table EMPLOYEE suivante pour stocker nos objets -

create table EMPLOYEE (
   id INT NOT NULL auto_increment,
   first_name VARCHAR(20) default NULL,
   last_name  VARCHAR(20) default NULL,
   salary     INT  default NULL,
   PRIMARY KEY (id)
);

Voici le mappage de la classe Employee avec des annotations pour mapper des objets avec la table EMPLOYEE définie -

import javax.persistence.*;

@Entity
@Table(name = "EMPLOYEE")
public class Employee {
   @Id @GeneratedValue
   @Column(name = "id")
   private int id;

   @Column(name = "first_name")
   private String firstName;

   @Column(name = "last_name")
   private String lastName;

   @Column(name = "salary")
   private int salary;  

   public Employee() {}
   
   public int getId() {
      return id;
   }
   
   public void setId( int id ) {
      this.id = id;
   }
   
   public String getFirstName() {
      return firstName;
   }
   
   public void setFirstName( String first_name ) {
      this.firstName = first_name;
   }
   
   public String getLastName() {
      return lastName;
   }
   
   public void setLastName( String last_name ) {
      this.lastName = last_name;
   }
   
   public int getSalary() {
      return salary;
   }
   
   public void setSalary( int salary ) {
      this.salary = salary;
   }
}

Hibernate détecte que l'annotation @Id se trouve sur un champ et suppose qu'elle doit accéder aux propriétés d'un objet directement via les champs au moment de l'exécution. Si vous avez placé l'annotation @Id sur la méthode getId (), vous activeriez l'accès aux propriétés via les méthodes getter et setter par défaut. Par conséquent, toutes les autres annotations sont également placées sur des champs ou sur des méthodes getter, suivant la stratégie sélectionnée.

La section suivante expliquera les annotations utilisées dans la classe ci-dessus.

Annotation @Entity

Les annotations standard EJB 3 sont contenues dans le javax.persistencepackage, nous importons donc ce package comme première étape. Deuxièmement, nous avons utilisé le@Entity annotation à la classe Employee, qui marque cette classe comme bean entité, elle doit donc avoir un constructeur sans argument visible avec au moins une portée protégée.

Annotation @Table

L'annotation @Table vous permet de spécifier les détails de la table qui sera utilisée pour conserver l'entité dans la base de données.

L'annotation @Table fournit quatre attributs, vous permettant de remplacer le nom de la table, son catalogue et son schéma, et d'appliquer des contraintes uniques sur les colonnes de la table. Pour l'instant, nous n'utilisons que le nom de la table, qui est EMPLOYEE.

Annotations @Id et @GeneratedValue

Chaque bean entité aura une clé primaire, que vous annotez sur la classe avec le @Idannotation. La clé primaire peut être un seul champ ou une combinaison de plusieurs champs selon la structure de votre table.

Par défaut, l'annotation @Id déterminera automatiquement la stratégie de génération de clé primaire la plus appropriée à utiliser, mais vous pouvez la remplacer en appliquant le @GeneratedValue annotation, qui prend deux paramètres strategy et generatordont je ne vais pas parler ici, alors utilisons uniquement la stratégie de génération de clé par défaut. Laisser Hibernate déterminer le type de générateur à utiliser rend votre code portable entre différentes bases de données.

Annotation @Column

L'annotation @Column est utilisée pour spécifier les détails de la colonne à laquelle un champ ou une propriété sera mappé. Vous pouvez utiliser l'annotation de colonne avec les attributs les plus couramment utilisés suivants:

  • name L'attribut permet de spécifier explicitement le nom de la colonne.

  • length L'attribut autorise la taille de la colonne utilisée pour mapper une valeur en particulier pour une valeur String.

  • nullable L'attribut permet à la colonne d'être marquée NOT NULL lorsque le schéma est généré.

  • unique L'attribut permet à la colonne d'être marquée comme contenant uniquement des valeurs uniques.

Créer une classe d'application

Enfin, nous allons créer notre classe d'application avec la méthode main () pour exécuter l'application. Nous utiliserons cette application pour sauvegarder quelques enregistrements d'employés, puis nous appliquerons les opérations CRUD sur ces enregistrements.

import java.util.List; 
import java.util.Date;
import java.util.Iterator; 
 
import org.hibernate.HibernateException; 
import org.hibernate.Session; 
import org.hibernate.Transaction;
import org.hibernate.cfg.AnnotationConfiguration;
import org.hibernate.SessionFactory;
import org.hibernate.cfg.Configuration;

public class ManageEmployee {
   private static SessionFactory factory; 
   public static void main(String[] args) {
      
      try {
         factory = new AnnotationConfiguration().
                   configure().
                   //addPackage("com.xyz") //add package if used.
                   addAnnotatedClass(Employee.class).
                   buildSessionFactory();
      } catch (Throwable ex) { 
         System.err.println("Failed to create sessionFactory object." + ex);
         throw new ExceptionInInitializerError(ex); 
      }
      
      ManageEmployee ME = new ManageEmployee();

      /* Add few employee records in database */
      Integer empID1 = ME.addEmployee("Zara", "Ali", 1000);
      Integer empID2 = ME.addEmployee("Daisy", "Das", 5000);
      Integer empID3 = ME.addEmployee("John", "Paul", 10000);

      /* List down all the employees */
      ME.listEmployees();

      /* Update employee's records */
      ME.updateEmployee(empID1, 5000);

      /* Delete an employee from the database */
      ME.deleteEmployee(empID2);

      /* List down new list of the employees */
      ME.listEmployees();
   }
   
   /* Method to CREATE an employee in the database */
   public Integer addEmployee(String fname, String lname, int salary){
      Session session = factory.openSession();
      Transaction tx = null;
      Integer employeeID = null;
      
      try {
         tx = session.beginTransaction();
         Employee employee = new Employee();
         employee.setFirstName(fname);
         employee.setLastName(lname);
         employee.setSalary(salary);
         employeeID = (Integer) session.save(employee); 
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
      return employeeID;
   }
   
   /* Method to  READ all the employees */
   public void listEmployees( ){
      Session session = factory.openSession();
      Transaction tx = null;
      
      try {
         tx = session.beginTransaction();
         List employees = session.createQuery("FROM Employee").list(); 
         for (Iterator iterator = employees.iterator(); iterator.hasNext();){
            Employee employee = (Employee) iterator.next(); 
            System.out.print("First Name: " + employee.getFirstName()); 
            System.out.print("  Last Name: " + employee.getLastName()); 
            System.out.println("  Salary: " + employee.getSalary()); 
         }
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
   }
   
   /* Method to UPDATE salary for an employee */
   public void updateEmployee(Integer EmployeeID, int salary ){
      Session session = factory.openSession();
      Transaction tx = null;
      
      try {
         tx = session.beginTransaction();
         Employee employee = (Employee)session.get(Employee.class, EmployeeID); 
         employee.setSalary( salary );
		 session.update(employee); 
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
   }
   
   /* Method to DELETE an employee from the records */
   public void deleteEmployee(Integer EmployeeID){
      Session session = factory.openSession();
      Transaction tx = null;
      
      try {
         tx = session.beginTransaction();
         Employee employee = (Employee)session.get(Employee.class, EmployeeID); 
         session.delete(employee); 
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
   }
}

Configuration de la base de données

Maintenant, créons hibernate.cfg.xml fichier de configuration pour définir les paramètres liés à la base de données.

<?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">
         cohondob
      </property>

   </session-factory>
</hibernate-configuration>

Compilation et exécution

Voici les étapes pour compiler et exécuter l'application mentionnée ci-dessus. Assurez-vous que vous avez correctement défini PATH et CLASSPATH avant de procéder à la compilation et à l'exécution.

  • Supprimez le fichier de mappage Employee.hbm.xml du chemin.

  • Créez le fichier source Employee.java comme indiqué ci-dessus et compilez-le.

  • Créez le fichier source ManageEmployee.java comme indiqué ci-dessus et compilez-le.

  • Exécutez le binaire ManageEmployee pour exécuter le programme.

Vous obtiendrez le résultat suivant et les enregistrements seraient créés dans la table EMPLOYEE.

$java ManageEmployee
.......VARIOUS LOG MESSAGES WILL DISPLAY HERE........

First Name: Zara  Last Name: Ali  Salary: 1000
First Name: Daisy  Last Name: Das  Salary: 5000
First Name: John  Last Name: Paul  Salary: 10000
First Name: Zara  Last Name: Ali  Salary: 5000
First Name: John  Last Name: Paul  Salary: 10000

Si vous vérifiez votre table EMPLOYEE, elle doit contenir les enregistrements suivants -

mysql> select * from EMPLOYEE;
+----+------------+-----------+--------+
| id | first_name | last_name | salary |
+----+------------+-----------+--------+
| 29 | Zara       | Ali       |   5000 |
| 31 | John       | Paul      |  10000 |
+----+------------+-----------+--------+
2 rows in set (0.00 sec

mysql>

Hibernate Query Language (HQL) est un langage de requête orienté objet, similaire à SQL, mais au lieu d'opérer sur des tables et des colonnes, HQL fonctionne avec des objets persistants et leurs propriétés. Les requêtes HQL sont traduites par Hibernate en requêtes SQL conventionnelles, qui à leur tour exécutent une action sur la base de données.

Bien que vous puissiez utiliser des instructions SQL directement avec Hibernate en utilisant Native SQL, je recommanderais d'utiliser HQL chaque fois que possible pour éviter les tracas de portabilité de la base de données et pour tirer parti des stratégies de génération et de mise en cache SQL d'Hibernate.

Les mots clés comme SELECT, FROM et WHERE, etc., ne sont pas sensibles à la casse, mais les propriétés telles que les noms de table et de colonne sont sensibles à la casse dans HQL.

Clause FROM

Vous utiliserez FROMclause si vous souhaitez charger un objet persistant complet en mémoire. Voici la syntaxe simple d'utilisation de la clause FROM -

String hql = "FROM Employee";
Query query = session.createQuery(hql);
List results = query.list();

Si vous avez besoin de qualifier complètement un nom de classe dans HQL, spécifiez simplement le nom du package et de la classe comme suit -

String hql = "FROM com.hibernatebook.criteria.Employee";
Query query = session.createQuery(hql);
List results = query.list();

Clause AS

le ASLa clause peut être utilisée pour attribuer des alias aux classes dans vos requêtes HQL, en particulier lorsque vous avez les requêtes longues. Par exemple, notre exemple simple précédent serait le suivant -

String hql = "FROM Employee AS E";
Query query = session.createQuery(hql);
List results = query.list();

le AS Le mot-clé est facultatif et vous pouvez également spécifier l'alias directement après le nom de la classe, comme suit -

String hql = "FROM Employee E";
Query query = session.createQuery(hql);
List results = query.list();

Clause SELECT

le SELECTLa clause fournit plus de contrôle sur l'ensemble de résultats que la clause from. Si vous souhaitez obtenir quelques propriétés d'objets au lieu de l'objet complet, utilisez la clause SELECT. Voici la syntaxe simple d'utilisation de la clause SELECT pour obtenir uniquement le champ first_name de l'objet Employee -

String hql = "SELECT E.firstName FROM Employee E";
Query query = session.createQuery(hql);
List results = query.list();

Il est à noter ici que Employee.firstName est une propriété de l'objet Employee plutôt qu'un champ de la table EMPLOYEE.

Clause WHERE

Si vous souhaitez restreindre les objets spécifiques renvoyés par le stockage, vous utilisez la clause WHERE. Voici la syntaxe simple d'utilisation de la clause WHERE -

String hql = "FROM Employee E WHERE E.id = 10";
Query query = session.createQuery(hql);
List results = query.list();

ORDER BY Clause

Pour trier les résultats de votre requête HQL, vous devrez utiliser le ORDER BYclause. Vous pouvez classer les résultats en fonction de n'importe quelle propriété des objets du jeu de résultats croissant (ASC) ou décroissant (DESC). Voici la syntaxe simple d'utilisation de la clause ORDER BY -

String hql = "FROM Employee E WHERE E.id > 10 ORDER BY E.salary DESC";
Query query = session.createQuery(hql);
List results = query.list();

Si vous vouliez trier par plus d'une propriété, vous ajouteriez simplement les propriétés supplémentaires à la fin de la clause order by, séparées par des virgules comme suit -

String hql = "FROM Employee E WHERE E.id > 10 " +
             "ORDER BY E.firstName DESC, E.salary DESC ";
Query query = session.createQuery(hql);
List results = query.list();

Clause GROUP BY

Cette clause permet à Hibernate d'extraire des informations de la base de données et de les regrouper en fonction de la valeur d'un attribut et, généralement, d'utiliser le résultat pour inclure une valeur agrégée. Voici la syntaxe simple d'utilisation de la clause GROUP BY -

String hql = "SELECT SUM(E.salary), E.firtName FROM Employee E " +
             "GROUP BY E.firstName";
Query query = session.createQuery(hql);
List results = query.list();

Utilisation des paramètres nommés

Hibernate prend en charge les paramètres nommés dans ses requêtes HQL. Cela facilite l'écriture de requêtes HQL qui acceptent les entrées de l'utilisateur et vous n'avez pas à vous défendre contre les attaques par injection SQL. Voici la syntaxe simple d'utilisation des paramètres nommés -

String hql = "FROM Employee E WHERE E.id = :employee_id";
Query query = session.createQuery(hql);
query.setParameter("employee_id",10);
List results = query.list();

Clause UPDATE

Les mises à jour en masse sont nouvelles pour HQL avec Hibernate 3 et les suppressions fonctionnent différemment dans Hibernate 3 et dans Hibernate 2. L'interface de requête contient désormais une méthode appelée executeUpdate () pour exécuter les instructions HQL UPDATE ou DELETE.

le UPDATELa clause peut être utilisée pour mettre à jour une ou plusieurs propriétés d'un ou plusieurs objets. Voici la syntaxe simple d'utilisation de la clause UPDATE -

String hql = "UPDATE Employee set salary = :salary "  + 
             "WHERE id = :employee_id";
Query query = session.createQuery(hql);
query.setParameter("salary", 1000);
query.setParameter("employee_id", 10);
int result = query.executeUpdate();
System.out.println("Rows affected: " + result);

Clause DELETE

le DELETELa clause peut être utilisée pour supprimer un ou plusieurs objets. Voici la syntaxe simple d'utilisation de la clause DELETE -

String hql = "DELETE FROM Employee "  + 
             "WHERE id = :employee_id";
Query query = session.createQuery(hql);
query.setParameter("employee_id", 10);
int result = query.executeUpdate();
System.out.println("Rows affected: " + result);

Clause INSERT

Prise en charge de HQL INSERT INTOclause uniquement où les enregistrements peuvent être insérés d'un objet à un autre objet. Voici la syntaxe simple d'utilisation de la clause INSERT INTO -

String hql = "INSERT INTO Employee(firstName, lastName, salary)"  + 
             "SELECT firstName, lastName, salary FROM old_employee";
Query query = session.createQuery(hql);
int result = query.executeUpdate();
System.out.println("Rows affected: " + result);

Méthodes agrégées

HQL prend en charge une gamme de méthodes d'agrégation, similaires à SQL. Ils fonctionnent de la même manière en HQL qu'en SQL et voici la liste des fonctions disponibles -

Sr.No. Fonctions et description
1

avg(property name)

La moyenne de la valeur d'une propriété

2

count(property name or *)

Le nombre de fois qu'une propriété apparaît dans les résultats

3

max(property name)

La valeur maximale des valeurs de propriété

4

min(property name)

La valeur minimale des valeurs de propriété

5

sum(property name)

La somme totale des valeurs de propriété

le distinctLe mot clé ne compte que les valeurs uniques de l'ensemble de lignes. La requête suivante renverra uniquement un nombre unique -

String hql = "SELECT count(distinct E.firstName) FROM Employee E";
Query query = session.createQuery(hql);
List results = query.list();

Pagination à l'aide de la requête

Il existe deux méthodes de l'interface de requête pour la pagination.

Sr.No. Méthode et description
1

Query setFirstResult(int startPosition)

Cette méthode prend un entier qui représente la première ligne de votre jeu de résultats, en commençant par la ligne 0.

2

Query setMaxResults(int maxResult)

Cette méthode indique à Hibernate de récupérer un nombre fixe maxResults d'objets.

En utilisant les deux méthodes ci-dessus ensemble, nous pouvons construire un composant de pagination dans notre application Web ou Swing. Voici l'exemple que vous pouvez étendre pour extraire 10 lignes à la fois -

String hql = "FROM Employee";
Query query = session.createQuery(hql);
query.setFirstResult(1);
query.setMaxResults(10);
List results = query.list();

Hibernate fournit d'autres moyens de manipuler les objets et à son tour les données disponibles dans les tables SGBDR. L'une des méthodes est l'API Criteria, qui vous permet de créer un objet de requête de critères par programme dans lequel vous pouvez appliquer des règles de filtrage et des conditions logiques.

L'Hibernate Session l'interface fournit createCriteria() méthode, qui peut être utilisée pour créer un Criteria objet qui renvoie des instances de la classe de l'objet de persistance lorsque votre application exécute une requête de critères.

Voici l'exemple le plus simple d'une requête de critères est un, qui retournera simplement chaque objet qui correspond à la classe Employee.

Criteria cr = session.createCriteria(Employee.class);
List results = cr.list();

Restrictions avec critères

Vous pouvez utiliser add() méthode disponible pour Criteriaobjet pour ajouter une restriction à une requête de critères. Voici l'exemple pour ajouter une restriction pour renvoyer les enregistrements avec un salaire égal à 2000 -

Criteria cr = session.createCriteria(Employee.class);
cr.add(Restrictions.eq("salary", 2000));
List results = cr.list();

Voici quelques exemples supplémentaires couvrant différents scénarios et peuvent être utilisés selon l'exigence -

Criteria cr = session.createCriteria(Employee.class);

// To get records having salary more than 2000
cr.add(Restrictions.gt("salary", 2000));

// To get records having salary less than 2000
cr.add(Restrictions.lt("salary", 2000));

// To get records having fistName starting with zara
cr.add(Restrictions.like("firstName", "zara%"));

// Case sensitive form of the above restriction.
cr.add(Restrictions.ilike("firstName", "zara%"));

// To get records having salary in between 1000 and 2000
cr.add(Restrictions.between("salary", 1000, 2000));

// To check if the given property is null
cr.add(Restrictions.isNull("salary"));

// To check if the given property is not null
cr.add(Restrictions.isNotNull("salary"));

// To check if the given property is empty
cr.add(Restrictions.isEmpty("salary"));

// To check if the given property is not empty
cr.add(Restrictions.isNotEmpty("salary"));

Vous pouvez créer des conditions AND ou OR en utilisant les restrictions LogicalExpression comme suit -

Criteria cr = session.createCriteria(Employee.class);

Criterion salary = Restrictions.gt("salary", 2000);
Criterion name = Restrictions.ilike("firstNname","zara%");

// To get records matching with OR conditions
LogicalExpression orExp = Restrictions.or(salary, name);
cr.add( orExp );

// To get records matching with AND conditions
LogicalExpression andExp = Restrictions.and(salary, name);
cr.add( andExp );

List results = cr.list();

Bien que toutes les conditions ci-dessus puissent être utilisées directement avec HQL, comme expliqué dans le tutoriel précédent.

Pagination à l'aide de critères

Il existe deux méthodes de l'interface Critères pour la pagination.

Sr.No. Méthode et description
1

public Criteria setFirstResult(int firstResult)

Cette méthode prend un entier qui représente la première ligne de votre jeu de résultats, en commençant par la ligne 0.

2

public Criteria setMaxResults(int maxResults)

Cette méthode indique à Hibernate de récupérer un nombre fixe maxResults d'objets.

En utilisant les deux méthodes ci-dessus ensemble, nous pouvons construire un composant de pagination dans notre application Web ou Swing. Voici l'exemple que vous pouvez étendre pour extraire 10 lignes à la fois -

Criteria cr = session.createCriteria(Employee.class);
cr.setFirstResult(1);
cr.setMaxResults(10);
List results = cr.list();

Trier les résultats

L'API Criteria fournit le org.hibernate.criterion.Orderclass pour trier votre jeu de résultats par ordre croissant ou décroissant, en fonction de l'une des propriétés de votre objet. Cet exemple montre comment vous utiliseriez la classe Order pour trier le jeu de résultats -

Criteria cr = session.createCriteria(Employee.class);

// To get records having salary more than 2000
cr.add(Restrictions.gt("salary", 2000));

// To sort records in descening order
cr.addOrder(Order.desc("salary"));

// To sort records in ascending order
cr.addOrder(Order.asc("salary"));

List results = cr.list();

Projections et agrégations

L'API Criteria fournit le org.hibernate.criterion.Projectionsclass, qui peut être utilisée pour obtenir la moyenne, le maximum ou le minimum des valeurs de propriété. La classe Projections est similaire à la classe Restrictions, en ce qu'elle fournit plusieurs méthodes de fabrique statiques pour obtenirProjection instances.

Voici les quelques exemples couvrant différents scénarios et peuvent être utilisés selon les besoins -

Criteria cr = session.createCriteria(Employee.class);

// To get total row count.
cr.setProjection(Projections.rowCount());

// To get average of a property.
cr.setProjection(Projections.avg("salary"));

// To get distinct count of a property.
cr.setProjection(Projections.countDistinct("firstName"));

// To get maximum of a property.
cr.setProjection(Projections.max("salary"));

// To get minimum of a property.
cr.setProjection(Projections.min("salary"));

// To get sum of a property.
cr.setProjection(Projections.sum("salary"));

Exemple de requêtes de critères

Considérez la classe POJO suivante -

public class Employee {
   private int id;
   private String firstName; 
   private String lastName;   
   private int salary;  

   public Employee() {}
   
   public Employee(String fname, String lname, int salary) {
      this.firstName = fname;
      this.lastName = lname;
      this.salary = salary;
   }
   
   public int getId() {
      return id;
   }
   
   public void setId( int id ) {
      this.id = id;
   }
   
   public String getFirstName() {
      return firstName;
   }
   
   public void setFirstName( String first_name ) {
      this.firstName = first_name;
   }
   
   public String getLastName() {
      return lastName;
   }
   
   public void setLastName( String last_name ) {
      this.lastName = last_name;
   }
   
   public int getSalary() {
      return salary;
   }
   
   public void setSalary( int salary ) {
      this.salary = salary;
   }
}

Créons la table EMPLOYEE suivante pour stocker les objets Employee -

create table EMPLOYEE (
   id INT NOT NULL auto_increment,
   first_name VARCHAR(20) default NULL,
   last_name  VARCHAR(20) default NULL,
   salary     INT  default NULL,
   PRIMARY KEY (id)
);

Voici le fichier de mappage.

<?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>
      
      <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>

Enfin, nous allons créer notre classe d'application avec la méthode main () pour exécuter l'application où nous allons utiliser Criteria requêtes -

import java.util.List; 
import java.util.Date;
import java.util.Iterator; 
 
import org.hibernate.HibernateException; 
import org.hibernate.Session; 
import org.hibernate.Transaction;
import org.hibernate.SessionFactory;
import org.hibernate.Criteria;
import org.hibernate.criterion.Restrictions;
import org.hibernate.criterion.Projections;
import org.hibernate.cfg.Configuration;

public class ManageEmployee {
   private static SessionFactory factory; 
   public static void main(String[] args) {
      
      try {
         factory = new Configuration().configure().buildSessionFactory();
      } catch (Throwable ex) { 
         System.err.println("Failed to create sessionFactory object." + ex);
         throw new ExceptionInInitializerError(ex); 
      }
      
      ManageEmployee ME = new ManageEmployee();

      /* Add few employee records in database */
      Integer empID1 = ME.addEmployee("Zara", "Ali", 2000);
      Integer empID2 = ME.addEmployee("Daisy", "Das", 5000);
      Integer empID3 = ME.addEmployee("John", "Paul", 5000);
      Integer empID4 = ME.addEmployee("Mohd", "Yasee", 3000);

      /* List down all the employees */
      ME.listEmployees();

      /* Print Total employee's count */
      ME.countEmployee();

      /* Print Total salary */
      ME.totalSalary();
   }
   
   /* Method to CREATE an employee in the database */
   public Integer addEmployee(String fname, String lname, int salary){
      Session session = factory.openSession();
      Transaction tx = null;
      Integer employeeID = null;
      
      try {
         tx = session.beginTransaction();
         Employee employee = new Employee(fname, lname, salary);
         employeeID = (Integer) session.save(employee); 
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
      return employeeID;
   }

   /* Method to  READ all the employees having salary more than 2000 */
   public void listEmployees( ) {
      Session session = factory.openSession();
      Transaction tx = null;
      
      try {
         tx = session.beginTransaction();
         Criteria cr = session.createCriteria(Employee.class);
         // Add restriction.
         cr.add(Restrictions.gt("salary", 2000));
         List employees = cr.list();

         for (Iterator iterator = employees.iterator(); iterator.hasNext();){
            Employee employee = (Employee) iterator.next(); 
            System.out.print("First Name: " + employee.getFirstName()); 
            System.out.print("  Last Name: " + employee.getLastName()); 
            System.out.println("  Salary: " + employee.getSalary()); 
         }
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
   }
   
   /* Method to print total number of records */
   public void countEmployee(){
      Session session = factory.openSession();
      Transaction tx = null;
      
      try {
         tx = session.beginTransaction();
         Criteria cr = session.createCriteria(Employee.class);

         // To get total row count.
         cr.setProjection(Projections.rowCount());
         List rowCount = cr.list();

         System.out.println("Total Coint: " + rowCount.get(0) );
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
   }
  
   /* Method to print sum of salaries */
   public void totalSalary(){
      Session session = factory.openSession();
      Transaction tx = null;
      
      try {
         tx = session.beginTransaction();
         Criteria cr = session.createCriteria(Employee.class);

         // To get total salary.
         cr.setProjection(Projections.sum("salary"));
         List totalSalary = cr.list();

         System.out.println("Total Salary: " + totalSalary.get(0) );
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
   }
}

Compilation et exécution

Voici les étapes pour compiler et exécuter l'application mentionnée ci-dessus. Assurez-vous que vous avez correctement défini PATH et CLASSPATH avant de procéder à la compilation et à l'exécution.

  • Créez le fichier de configuration hibernate.cfg.xml comme expliqué dans le chapitre de configuration.

  • Créez le fichier de mappage Employee.hbm.xml comme indiqué ci-dessus.

  • Créez le fichier source Employee.java comme indiqué ci-dessus et compilez-le.

  • Créez le fichier source ManageEmployee.java comme indiqué ci-dessus et compilez-le.

  • Exécutez le binaire ManageEmployee pour exécuter le programme.

Vous obtiendrez le résultat suivant et les enregistrements seraient créés dans la table EMPLOYEE.

$java ManageEmployee
.......VARIOUS LOG MESSAGES WILL DISPLAY HERE........

First Name: Daisy  Last Name: Das  Salary: 5000
First Name: John  Last Name: Paul  Salary: 5000
First Name: Mohd  Last Name: Yasee  Salary: 3000
Total Coint: 4
Total Salary: 15000

Si vous vérifiez votre table EMPLOYEE, elle doit contenir les enregistrements suivants:

mysql> select * from EMPLOYEE;
+----+------------+-----------+--------+
| id | first_name | last_name | salary |
+----+------------+-----------+--------+
| 14 | Zara       | Ali       |   2000 |
| 15 | Daisy      | Das       |   5000 |
| 16 | John       | Paul      |   5000 |
| 17 | Mohd       | Yasee     |   3000 |
+----+------------+-----------+--------+
4 rows in set (0.00 sec)
mysql>

Vous pouvez utiliser SQL natif pour exprimer des requêtes de base de données si vous souhaitez utiliser des fonctionnalités spécifiques à la base de données, telles que des conseils de requête ou le mot clé CONNECT dans Oracle. Hibernate 3.x vous permet de spécifier du SQL manuscrit, y compris des procédures stockées, pour toutes les opérations de création, de mise à jour, de suppression et de chargement.

Votre application créera une requête SQL native à partir de la session avec le createSQLQuery() méthode sur l'interface Session -

public SQLQuery createSQLQuery(String sqlString) throws HibernateException

Après avoir transmis une chaîne contenant la requête SQL à la méthode createSQLQuery (), vous pouvez associer le résultat SQL à une entité Hibernate existante, une jointure ou un résultat scalaire à l'aide des méthodes addEntity (), addJoin () et addScalar () respectivement.

Requêtes scalaires

La requête SQL la plus basique consiste à obtenir une liste de scalaires (valeurs) à partir d'une ou plusieurs tables. Voici la syntaxe d'utilisation du SQL natif pour les valeurs scalaires -

String sql = "SELECT first_name, salary FROM EMPLOYEE";
SQLQuery query = session.createSQLQuery(sql);
query.setResultTransformer(Criteria.ALIAS_TO_ENTITY_MAP);
List results = query.list();

Requêtes d'entité

Les requêtes ci-dessus concernaient uniquement le renvoi de valeurs scalaires, renvoyant essentiellement les valeurs «brutes» du jeu de résultats. Voici la syntaxe pour obtenir les objets d'entité dans leur ensemble à partir d'une requête SQL native via addEntity ().

String sql = "SELECT * FROM EMPLOYEE";
SQLQuery query = session.createSQLQuery(sql);
query.addEntity(Employee.class);
List results = query.list();

Requêtes SQL nommées

Voici la syntaxe pour obtenir des objets d'entité à partir d'une requête SQL native via addEntity () et en utilisant une requête SQL nommée.

String sql = "SELECT * FROM EMPLOYEE WHERE id = :employee_id";
SQLQuery query = session.createSQLQuery(sql);
query.addEntity(Employee.class);
query.setParameter("employee_id", 10);
List results = query.list();

Exemple SQL natif

Considérez la classe POJO suivante -

public class Employee {
   private int id;
   private String firstName; 
   private String lastName;   
   private int salary;  

   public Employee() {}
   
   public Employee(String fname, String lname, int salary) {
      this.firstName = fname;
      this.lastName = lname;
      this.salary = salary;
   }
   
   public int getId() {
      return id;
   }
   
   public void setId( int id ) {
      this.id = id;
   }
   
   public String getFirstName() {
      return firstName;
   }
   
   public void setFirstName( String first_name ) {
      this.firstName = first_name;
   }
   
   public String getLastName() {
      return lastName;
   }
   
   public void setLastName( String last_name ) {
      this.lastName = last_name;
   }
   
   public int getSalary() {
      return salary;
   }
   
   public void setSalary( int salary ) {
      this.salary = salary;
   }
}

Créons la table EMPLOYEE suivante pour stocker les objets Employee -

create table EMPLOYEE (
   id INT NOT NULL auto_increment,
   first_name VARCHAR(20) default NULL,
   last_name  VARCHAR(20) default NULL,
   salary     INT  default NULL,
   PRIMARY KEY (id)
);

Voici le fichier de mappage -

<?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>
      
      <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>

Enfin, nous allons créer notre classe d'application avec la méthode main () pour exécuter l'application où nous allons utiliser Native SQL requêtes -

import java.util.*; 
 
import org.hibernate.HibernateException; 
import org.hibernate.Session; 
import org.hibernate.Transaction;
import org.hibernate.SessionFactory;
import org.hibernate.SQLQuery;
import org.hibernate.Criteria;
import org.hibernate.Hibernate;
import org.hibernate.cfg.Configuration;

public class ManageEmployee {
   private static SessionFactory factory; 
   public static void main(String[] args) {
      
      try {
         factory = new Configuration().configure().buildSessionFactory();
      } catch (Throwable ex) { 
         System.err.println("Failed to create sessionFactory object." + ex);
         throw new ExceptionInInitializerError(ex); 
      }
      
      ManageEmployee ME = new ManageEmployee();

      /* Add few employee records in database */
      Integer empID1 = ME.addEmployee("Zara", "Ali", 2000);
      Integer empID2 = ME.addEmployee("Daisy", "Das", 5000);
      Integer empID3 = ME.addEmployee("John", "Paul", 5000);
      Integer empID4 = ME.addEmployee("Mohd", "Yasee", 3000);

      /* List down employees and their salary using Scalar Query */
      ME.listEmployeesScalar();

      /* List down complete employees information using Entity Query */
      ME.listEmployeesEntity();
   }
   
   /* Method to CREATE an employee in the database */
   public Integer addEmployee(String fname, String lname, int salary){
      Session session = factory.openSession();
      Transaction tx = null;
      Integer employeeID = null;
      
      try {
         tx = session.beginTransaction();
         Employee employee = new Employee(fname, lname, salary);
         employeeID = (Integer) session.save(employee); 
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
      return employeeID;
   }

   /* Method to  READ all the employees using Scalar Query */
   public void listEmployeesScalar( ){
      Session session = factory.openSession();
      Transaction tx = null;
      
      try {
         tx = session.beginTransaction();
         String sql = "SELECT first_name, salary FROM EMPLOYEE";
         SQLQuery query = session.createSQLQuery(sql);
         query.setResultTransformer(Criteria.ALIAS_TO_ENTITY_MAP);
         List data = query.list();

         for(Object object : data) {
            Map row = (Map)object;
            System.out.print("First Name: " + row.get("first_name")); 
            System.out.println(", Salary: " + row.get("salary")); 
         }
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
   }

   /* Method to READ all the employees using Entity Query */
   public void listEmployeesEntity( ){
      Session session = factory.openSession();
      Transaction tx = null;
      
      try {
         tx = session.beginTransaction();
         String sql = "SELECT * FROM EMPLOYEE";
         SQLQuery query = session.createSQLQuery(sql);
         query.addEntity(Employee.class);
         List employees = query.list();

         for (Iterator iterator = employees.iterator(); iterator.hasNext();){
            Employee employee = (Employee) iterator.next(); 
            System.out.print("First Name: " + employee.getFirstName()); 
            System.out.print("  Last Name: " + employee.getLastName()); 
            System.out.println("  Salary: " + employee.getSalary()); 
         }
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
   }
}

Compilation et exécution

Voici les étapes pour compiler et exécuter l'application mentionnée ci-dessus. Assurez-vous que vous avez correctement défini PATH et CLASSPATH avant de procéder à la compilation et à l'exécution.

  • Créez le fichier de configuration hibernate.cfg.xml comme expliqué dans le chapitre de configuration.

  • Créez le fichier de mappage Employee.hbm.xml comme indiqué ci-dessus.

  • Créez le fichier source Employee.java comme indiqué ci-dessus et compilez-le.

  • Créez le fichier source ManageEmployee.java comme indiqué ci-dessus et compilez-le.

  • Exécutez le binaire ManageEmployee pour exécuter le programme.

Vous obtiendrez le résultat suivant et les enregistrements seraient créés dans la table EMPLOYEE.

$java ManageEmployee
.......VARIOUS LOG MESSAGES WILL DISPLAY HERE........

First Name: Zara, Salary: 2000
First Name: Daisy, Salary: 5000
First Name: John, Salary: 5000
First Name: Mohd, Salary: 3000
First Name: Zara  Last Name: Ali  Salary: 2000
First Name: Daisy  Last Name: Das  Salary: 5000
First Name: John  Last Name: Paul  Salary: 5000
First Name: Mohd  Last Name: Yasee  Salary: 3000

Si vous vérifiez votre table EMPLOYEE, elle doit contenir les enregistrements suivants -

mysql> select * from EMPLOYEE;
+----+------------+-----------+--------+
| id | first_name | last_name | salary |
+----+------------+-----------+--------+
| 26 | Zara       | Ali       |   2000 |
| 27 | Daisy      | Das       |   5000 |
| 28 | John       | Paul      |   5000 |
| 29 | Mohd       | Yasee     |   3000 |
+----+------------+-----------+--------+
4 rows in set (0.00 sec)
mysql>

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 mis 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.

Sr.No. Nom et description du cache
1

EHCache

Il peut mettre en cache en mémoire ou sur disque et en cluster et il prend en charge le cache de résultats 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 JVM 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 maintenant 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 est inutile d'activer un 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 un support de cache très fin 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.

Considérez une situation où vous devez télécharger un grand nombre d'enregistrements dans votre base de données à l'aide d'Hibernate. Voici l'extrait de code pour y parvenir en utilisant Hibernate -

Session session = SessionFactory.openSession();
Transaction tx = session.beginTransaction();
for ( int i=0; i<100000; i++ ) {
   Employee employee = new Employee(.....);
   session.save(employee);
}
tx.commit();
session.close();

Par défaut, Hibernate mettra en cache tous les objets persistants dans le cache de niveau session et finalement votre application tomberait avec un OutOfMemoryExceptionquelque part autour de la 50 000e rangée. Vous pouvez résoudre ce problème si vous utilisezbatch processing avec Hibernate.

Pour utiliser la fonction de traitement par lots, définissez d'abord hibernate.jdbc.batch_sizecomme taille de lot à un nombre à 20 ou 50 selon la taille de l'objet. Cela indiquera au conteneur d'hibernation que toutes les X lignes doivent être insérées en tant que lot. Pour implémenter cela dans votre code, nous aurions besoin de faire peu de modifications comme suit -

Session session = SessionFactory.openSession();
Transaction tx = session.beginTransaction();
for ( int i=0; i<100000; i++ ) {
   Employee employee = new Employee(.....);
   session.save(employee);
   if( i % 50 == 0 ) { // Same as the JDBC batch size
      //flush a batch of inserts and release memory:
      session.flush();
      session.clear();
   }
}
tx.commit();
session.close();

Le code ci-dessus fonctionnera bien pour l'opération INSERT, mais si vous êtes prêt à effectuer l'opération UPDATE, vous pouvez le faire en utilisant le code suivant -

Session session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();

ScrollableResults employeeCursor = session.createQuery("FROM EMPLOYEE").scroll();
int count = 0;

while ( employeeCursor.next() ) {
   Employee employee = (Employee) employeeCursor.get(0);
   employee.updateEmployee();
   seession.update(employee); 
   if ( ++count % 50 == 0 ) {
      session.flush();
      session.clear();
   }
}
tx.commit();
session.close();

Exemple de traitement par lots

Modifions le fichier de configuration pour ajouter hibernate.jdbc.batch_size propriété -

<?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.jdbc.batch_size">
         50
      </property>

      <!-- List of XML mapping files -->
      <mapping resource = "Employee.hbm.xml"/>

   </session-factory>
</hibernate-configuration>

Considérez la classe d'employé POJO suivante -

public class Employee {
   private int id;
   private String firstName; 
   private String lastName;   
   private int salary;  

   public Employee() {}
   
   public Employee(String fname, String lname, int salary) {
      this.firstName = fname;
      this.lastName = lname;
      this.salary = salary;
   }
   
   public int getId() {
      return id;
   }
   
   public void setId( int id ) {
      this.id = id;
   }
   
   public String getFirstName() {
      return firstName;
   }
   
   public void setFirstName( String first_name ) {
      this.firstName = first_name;
   }
   
   public String getLastName() {
      return lastName;
   }
   
   public void setLastName( String last_name ) {
      this.lastName = last_name;
   }
   
   public int getSalary() {
      return salary;
   }
   
   public void setSalary( int salary ) {
      this.salary = salary;
   }
}

Créons la table EMPLOYEE suivante pour stocker les objets Employee -

create table EMPLOYEE (
   id INT NOT NULL auto_increment,
   first_name VARCHAR(20) default NULL,
   last_name  VARCHAR(20) default NULL,
   salary     INT  default NULL,
   PRIMARY KEY (id)
);

Voici le fichier de mappage pour mapper les objets Employé avec la table EMPLOYEE -

<?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>
      
      <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>

Enfin, nous allons créer notre classe d'application avec la méthode main () pour exécuter l'application où nous allons utiliser flush() et clear() méthodes disponibles avec l'objet Session afin qu'Hibernate continue d'écrire ces enregistrements dans la base de données au lieu de les mettre en cache dans la mémoire.

import java.util.*; 
 
import org.hibernate.HibernateException; 
import org.hibernate.Session; 
import org.hibernate.Transaction;
import org.hibernate.SessionFactory;
import org.hibernate.cfg.Configuration;

public class ManageEmployee {
   private static SessionFactory factory; 
   public static void main(String[] args) {
      
      try {
         factory = new Configuration().configure().buildSessionFactory();
      } catch (Throwable ex) { 
         System.err.println("Failed to create sessionFactory object." + ex);
         throw new ExceptionInInitializerError(ex); 
      }
      ManageEmployee ME = new ManageEmployee();

      /* Add employee records in batches */
      ME.addEmployees( );
   }
   
   /* Method to create employee records in batches */
   public void addEmployees( ){
      Session session = factory.openSession();
      Transaction tx = null;
      Integer employeeID = null;
      
      try {
         tx = session.beginTransaction();
         for ( int i=0; i<100000; i++ ) {
            String fname = "First Name " + i;
            String lname = "Last Name " + i;
            Integer salary = i;
            Employee employee = new Employee(fname, lname, salary);
            session.save(employee);
         	if( i % 50 == 0 ) {
               session.flush();
               session.clear();
            }
         }
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
      return ;
   }
}

Compilation et exécution

Voici les étapes pour compiler et exécuter l'application mentionnée ci-dessus. Assurez-vous que vous avez correctement défini PATH et CLASSPATH avant de procéder à la compilation et à l'exécution.

  • Créez le fichier de configuration hibernate.cfg.xml comme expliqué ci-dessus.

  • Créez le fichier de mappage Employee.hbm.xml comme indiqué ci-dessus.

  • Créez le fichier source Employee.java comme indiqué ci-dessus et compilez-le.

  • Créez le fichier source ManageEmployee.java comme indiqué ci-dessus et compilez-le.

  • Exécutez le binaire ManageEmployee pour exécuter le programme, qui créera 100 000 enregistrements dans la table EMPLOYEE.

Comme vous l'avez appris, dans Hibernate, un objet sera créé et conservé. Une fois que l'objet a été modifié, il doit être réenregistré dans la base de données. Ce processus se poursuit jusqu'à la prochaine fois que l'objet est nécessaire, et il sera chargé à partir du magasin persistant.

Ainsi, un objet passe par différentes étapes de son cycle de vie et Interceptor Interfacefournit des méthodes qui peuvent être appelées à différentes étapes pour effectuer certaines tâches requises. Ces méthodes sont des rappels de la session à l'application, permettant à l'application d'inspecter et / ou de manipuler les propriétés d'un objet persistant avant qu'il ne soit enregistré, mis à jour, supprimé ou chargé. Voici la liste de toutes les méthodes disponibles dans l'interface Interceptor -

Sr.No. Méthode et description
1

findDirty()

Cette méthode est appelée lorsque le flush() est appelée sur un objet Session.

2

instantiate()

Cette méthode est appelée lorsqu'une classe persistante est instanciée.

3

isUnsaved()

Cette méthode est appelée lorsqu'un objet est passé au saveOrUpdate() méthode/

4

onDelete()

Cette méthode est appelée avant la suppression d'un objet.

5

onFlushDirty()

Cette méthode est appelée lorsque Hibernate détecte qu'un objet est sale (c'est-à-dire qu'il a été modifié) pendant une opération de vidage, c'est-à-dire de mise à jour.

6

onLoad()

Cette méthode est appelée avant l'initialisation d'un objet.

sept

onSave()

Cette méthode est appelée avant l'enregistrement d'un objet.

8

postFlush()

Cette méthode est appelée après un vidage et un objet a été mis à jour en mémoire.

9

preFlush()

Cette méthode est appelée avant un flush.

Hibernate Interceptor nous donne un contrôle total sur l'apparence d'un objet à la fois pour l'application et la base de données.

Comment utiliser les intercepteurs?

Pour construire un intercepteur, vous pouvez soit implémenter Interceptor classe directement ou étendre EmptyInterceptorclasse. Voici les étapes simples pour utiliser la fonctionnalité Hibernate Interceptor.

Créer des intercepteurs

Nous allons étendre EmptyInterceptor dans notre exemple où la méthode d'Interceptor sera appelée automatiquement lorsque Employeel'objet est créé et mis à jour. Vous pouvez implémenter plus de méthodes selon vos besoins.

import java.io.Serializable;
import java.util.Date;
import java.util.Iterator;

import org.hibernate.EmptyInterceptor;
import org.hibernate.Transaction;
import org.hibernate.type.Type;

public class MyInterceptor extends EmptyInterceptor {
   private int updates;
   private int creates;
   private int loads;

   public void onDelete(Object entity, Serializable id,
      Object[] state, String[] propertyNames, Type[] types) {
       // do nothing
   }

   // This method is called when Employee object gets updated.
   public boolean onFlushDirty(Object entity, Serializable id,
      Object[] currentState, Object[] previousState, String[] propertyNames,
      Type[] types) {
         if ( entity instanceof Employee ) {
            System.out.println("Update Operation");
            return true; 
         }
         return false;
   }
	
   public boolean onLoad(Object entity, Serializable id,
      Object[] state, String[] propertyNames, Type[] types) {
         // do nothing
         return true;
   }
   
   // This method is called when Employee object gets created.
   public boolean onSave(Object entity, Serializable id,
      Object[] state, String[] propertyNames, Type[] types) {
         if ( entity instanceof Employee ) {
            System.out.println("Create Operation");
            return true; 
         }
         return false;
   }
   
   //called before commit into database
   public void preFlush(Iterator iterator) {
      System.out.println("preFlush");
   }
   
   //called after committed into database
   public void postFlush(Iterator iterator) {
      System.out.println("postFlush");
   }
}

Créer des classes POJO

Maintenant, modifions un peu notre premier exemple où nous avons utilisé la table EMPLOYEE et la classe Employee pour jouer avec -

public class Employee {
   private int id;
   private String firstName; 
   private String lastName;   
   private int salary;  

   public Employee() {}
   
   public Employee(String fname, String lname, int salary) {
      this.firstName = fname;
      this.lastName = lname;
      this.salary = salary;
   }
   
   public int getId() {
      return id;
   }
   
   public void setId( int id ) {
      this.id = id;
   }
   
   public String getFirstName() {
      return firstName;
   }
   
   public void setFirstName( String first_name ) {
      this.firstName = first_name;
   }
   
   public String getLastName() {
      return lastName;
   }
   
   public void setLastName( String last_name ) {
      this.lastName = last_name;
   }
   
   public int getSalary() {
      return salary;
   }
   
   public void setSalary( int salary ) {
      this.salary = salary;
   }
}

Créer des tables de base de données

La deuxième étape serait de créer des tables dans votre base de données. Il y aurait une table correspondant à chaque objet, vous êtes prêt à fournir la persistance. Considérez les objets expliqués ci-dessus, doivent être stockés et récupérés dans le tableau SGBDR suivant -

create table EMPLOYEE (
   id INT NOT NULL auto_increment,
   first_name VARCHAR(20) default NULL,
   last_name  VARCHAR(20) default NULL,
   salary     INT  default NULL,
   PRIMARY KEY (id)
);

Créer un fichier de configuration de mappage

Cette étape consiste à créer un fichier de mappage qui indique à Hibernate - comment mapper la ou les classes définies aux tables de la base de données.

<?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>
      
      <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>

Créer une classe d'application

Enfin, nous allons créer notre classe d'application avec la méthode main () pour exécuter l'application. Ici, il convient de noter que lors de la création d'un objet de session, nous avons utilisé notre classe Interceptor comme argument.

import java.util.List; 
import java.util.Date;
import java.util.Iterator; 
 
import org.hibernate.HibernateException; 
import org.hibernate.Session; 
import org.hibernate.Transaction;
import org.hibernate.SessionFactory;
import org.hibernate.cfg.Configuration;

public class ManageEmployee {
   private static SessionFactory factory; 
   public static void main(String[] args) {
      
      try {
         factory = new Configuration().configure().buildSessionFactory();
      } catch (Throwable ex) { 
         System.err.println("Failed to create sessionFactory object." + ex);
         throw new ExceptionInInitializerError(ex); 
      }

      ManageEmployee ME = new ManageEmployee();

      /* Add few employee records in database */
      Integer empID1 = ME.addEmployee("Zara", "Ali", 1000);
      Integer empID2 = ME.addEmployee("Daisy", "Das", 5000);
      Integer empID3 = ME.addEmployee("John", "Paul", 10000);

      /* List down all the employees */
      ME.listEmployees();

      /* Update employee's records */
      ME.updateEmployee(empID1, 5000);

      /* Delete an employee from the database */
      ME.deleteEmployee(empID2);

      /* List down new list of the employees */
      ME.listEmployees();
   }
   
   /* Method to CREATE an employee in the database */
   public Integer addEmployee(String fname, String lname, int salary){
      Session session = factory.openSession( new MyInterceptor() );
      Transaction tx = null;
      Integer employeeID = null;
      
      try {
         tx = session.beginTransaction();
         Employee employee = new Employee(fname, lname, salary);
         employeeID = (Integer) session.save(employee); 
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
      return employeeID;
   }
   
   /* Method to  READ all the employees */
   public void listEmployees( ){
      Session session = factory.openSession( new MyInterceptor() );
      Transaction tx = null;
      
      try {
         tx = session.beginTransaction();
         List employees = session.createQuery("FROM Employee").list(); 
         for (Iterator iterator = employees.iterator(); iterator.hasNext();){
            Employee employee = (Employee) iterator.next(); 
            System.out.print("First Name: " + employee.getFirstName()); 
            System.out.print("  Last Name: " + employee.getLastName()); 
            System.out.println("  Salary: " + employee.getSalary()); 
         }
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
   }
   
   /* Method to UPDATE salary for an employee */
   public void updateEmployee(Integer EmployeeID, int salary ){
      Session session = factory.openSession( new MyInterceptor() );
      Transaction tx = null;
      
      try {
         tx = session.beginTransaction();
         Employee employee = (Employee)session.get(Employee.class, EmployeeID); 
         employee.setSalary( salary );
		 session.update(employee); 
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
   }
   
   /* Method to DELETE an employee from the records */
   public void deleteEmployee(Integer EmployeeID){
      Session session = factory.openSession( new MyInterceptor() );
      Transaction tx = null;
      
      try {
         tx = session.beginTransaction();
         Employee employee = (Employee)session.get(Employee.class, EmployeeID); 
         session.delete(employee); 
         tx.commit();
      } catch (HibernateException e) {
         if (tx!=null) tx.rollback();
         e.printStackTrace(); 
      } finally {
         session.close(); 
      }
   }
}

Compilation et exécution

Voici les étapes pour compiler et exécuter l'application mentionnée ci-dessus. Assurez-vous que vous avez correctement défini PATH et CLASSPATH avant de procéder à la compilation et à l'exécution.

  • Créez le fichier de configuration hibernate.cfg.xml comme expliqué dans le chapitre de configuration.

  • Créez le fichier de mappage Employee.hbm.xml comme indiqué ci-dessus.

  • Créez le fichier source Employee.java comme indiqué ci-dessus et compilez-le.

  • Créez le fichier source MyInterceptor.java comme indiqué ci-dessus et compilez-le.

  • Créez le fichier source ManageEmployee.java comme indiqué ci-dessus et compilez-le.

  • Exécutez le binaire ManageEmployee pour exécuter le programme.

Vous obtiendrez le résultat suivant et les enregistrements seraient créés dans la table EMPLOYEE.

$java ManageEmployee
.......VARIOUS LOG MESSAGES WILL DISPLAY HERE........

Create Operation
preFlush
postFlush
Create Operation
preFlush
postFlush
Create Operation
preFlush
postFlush
First Name: Zara  Last Name: Ali  Salary: 1000
First Name: Daisy  Last Name: Das  Salary: 5000
First Name: John  Last Name: Paul  Salary: 10000
preFlush
postFlush
preFlush
Update Operation
postFlush
preFlush
postFlush
First Name: Zara  Last Name: Ali  Salary: 5000
First Name: John  Last Name: Paul  Salary: 10000
preFlush
postFlush

Si vous vérifiez votre table EMPLOYEE, elle doit contenir les enregistrements suivants -

mysql> select * from EMPLOYEE;
+----+------------+-----------+--------+
| id | first_name | last_name | salary |
+----+------------+-----------+--------+
| 29 | Zara       | Ali       |   5000 |
| 31 | John       | Paul      |  10000 |
+----+------------+-----------+--------+
2 rows in set (0.00 sec
mysql>