Hibernate - Intercepteurs
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 -
N ° Sr. | 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>