17 Hibernate3 Gestione efficiente

Hibernate
Modificare gli oggetti
efficientemente
Dott. Doria Mauro
doriamauro@gmail.com
doriamauro@gmail.com
Transitive persistence
La persistenza transitiva è una tecnica che consente di
propagare automaticamente la persistenza ad un sottografico di
oggetti transienti e/o detached.
Ad esempio, nell’associazione tra Item e Bid, è possibile legare il
ciclo di vita di un oggetto bid a quello dell’ item in cui si trova:
–
–
2
Un bid viene automaticamente reso persistente al momento in cui
rendiamo persistente l’item
Un bid viene cancellato al momento che cancelliamo un item.
Esistono diversi modi per realiazzare la transitive persistence,
anche se si basano tutti sugli stessi principi
doriamauro@gmail.com
Persistence by reachability
3
Uno strato di persistenza si dice che implementa la persistence
by reachability se una istanza diventa persistente ogni volta che
l’applicazione crea un reference a questa istanza da un’altra
istanza che è già persistente.
doriamauro@gmail.com
Persistence by reachability
Nell’esempio precedente:
–
–
–
4
Quando l’oggetto Computer diventa persistente, lo diventano pure
Desktop PCs e Monitor
Quando Computer diventa detached lo diventano anche Desktop
PCs e Monitor
Se uno di questi due oggetti perde il reference da Computer, diventa
automaticamente transient (e quindi rimosso).
Si presume una navigazione nel grafo degli oggetti unidirezionale
da padre a figlio.
Navigando nel grafo degli oggetti, si “incontrano” soltanto oggetti
persistenti; non sono mai possibili violazioni di integrità
referenziale.
doriamauro@gmail.com
Persistence by reachability
Per quanto possa sembrare una strategia affascinante, questa
tecnica non è applicabile così com’è (ne da Hibernate, ne da altri
ORM framework)
Sarebbe necessario una sorta di garbage collector per la
persistenza con un comportamento simile a quello della JVM
Il problema più grosso sta nel fatto che:
–
–
5
Gli oggetti sono condivisi (quindi ad un oggetto possono essere
collegati references da diversi altri oggetti)
Gli oggetti presenti nel persistent context non rappresentano tutti i
legami presenti nel DB
Per fare questo G.C. sarebbe necessario fare una costante
scansione del DB alla ricerca dei legami tra le righe
doriamauro@gmail.com
Persistenza a cascata
6
Hibernate applica un tipo più semplice di persistence by
reachability: la persistenza a cascata.
Hibernate propaga lo stato di un oggetto verso un altro
referenziato in base a quanto descritto nella proprietà “cascade”.
La persistenza a cascata è più flessibile e assegna maggiore
controllo allo sviluppatore.
Solo così è possibile sapere facilmente quando un oggetto diventa detached.
Non si attiva di default ed è disponibile in tutti i tipi di
associazione: (uno-a-uno, uno-a-molti, molti-a-molti) e per tutte le
collections (<set>, <bag>, <list> e <map>).
doriamauro@gmail.com
Persistenza a cascata
Tutti i valori per l’attributo cascade:
(Come save-update)
7
doriamauro@gmail.com
Persistenza a cascata
8
doriamauro@gmail.com
Persistenza a cascata
9
Il valore delete-orphan si può applicare soltanto alle collections
doriamauro@gmail.com
Persistenza a cascata
E’ possibile combinare tra loro diverse opzioni. Ecco alcuni
esempi:
<many-to-one name="parent"
column="PARENT_CATEGORY_ID"
class="Category"
cascade="save-update, persist, merge"/>
NOTA: il valore delete-orphan
non è incluso quando si
sceglie il valore all
<set name="bids" cascade="all, delete-orphan"
inverse="true">
<key column ="ITEM_ID"/>
<one-to-many class="Bid"/>
</set>
NOTA: non vi è nessuna relazione tra l’attributo cascade e quello inverse
10
doriamauro@gmail.com
Esempi di cascade
11
Un amministratore deve poter aggiungere, eliminare e modificare
tutte le categorie dei prodotti che vende.
Il mapping della classe Category:
<class name="Category" table="CATEGORY">
<property name="name" column="CATEGORY_NAME"/>
<many-to-one name="parentCategory"
class="Category"
column="PARENT_CATEGORY_ID"
cascade="none"/>
<set name="childCategories"
table="CATEGORY"
cascade="save-update"
inverse="true">
<key column="PARENT_CATEGORY_ID"/>
<one-to-many class="Category"/>
</set>
...
NOTA: si tratta di una
associazione
autoreferenziata di tipo
molti-a-uno
doriamauro@gmail.com
Esempi di cascade
aggiungere una nuova categoria
E’ possibile aggiungere una categoria in diversi modi:
Session session = sessionFactory.openSession();
L’oggetto computer
Transaction tx = session.beginTransaction();
diventa persistent
Category computer =
(Category) session.load(Category.class, computerId);
Category laptops = new Category("Laptops");
computer.getChildCategories().add(laptops);
laptops.setParentCategory(computer);
tx.commit();
session.close();
Completa la
bidirezionalità
dell’associazione
12
L’oggetto laptop
diventa persistente
sotto l’effetto del
cascade
doriamauro@gmail.com
Esempi di cascade
aggiungere una nuova categoria: modalità detached
La stessa operazione si può fare quando la categoria padre è
detached:
Queste operazioni in nessuna sessione:
Category computer = (Category) session.get(…) // Caricato in una precedente sessione
Category laptops = new Category("Laptops");
computer.getChildCategories().add(laptops);
laptops.setParentCategory(computer);
Queste operazioni in una seconda sessione:
Cosa succederebbe se
il cascade fosse attivato
per l’associazione
<many-to-one>?
Session session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
// Rendere persistente la nuova categoria già linkata a suo padre
session.save(laptops);
tx.commit();
Hibernate valuta il link verso il padre e aggiunge
session.close();
13
NO CASCADE QUI!!!
il riferimento alla riga padre anche nel DB
indipendentemente da cascade e inverse
doriamauro@gmail.com
Esempi di cascade
aggiungere una nuova categoria: modalità detached
Vediamo un altro esempio per valutare la convenienza di questo
stile detached:
Queste operazioni in nessuna sessione:
Category computer = (Category) session.get(…) // Caricato in una precedente sessione
Category laptops = new Category("Laptops");
Metodi che creano
Category laptopUltraPortable = new Category("Ultra-Portable");
l’associazione in tutte
Category laptopTabletPCs = new Category("Tablet PCs");
laptops.addChildCategory(laptopUltraPortable);
e due i sensi
laptops.addChildCategory(laptopTabletPCs);
computer.addChildCategory(laptops);
14
Queste operazioni in una seconda sessione:
Session session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
// Rendere persistente tutte le categorie
session.save(laptops);
tx.commit();
session.close();
Hibernate valuta il link verso il padre e
aggiunge il riferimento alla riga padre
anche nel DB e poi salva laptop e,a
cascata, tutte le sue sottocategorie
doriamauro@gmail.com
Esempi di cascade
aggiungere una nuova categoria: modalità detached
Vediamo ancora un altro esempio per valutare la capacità di
updatare (oltre che salvare) a cascata:
Queste, dopo aver chiuso la sessione dell’esempio precedente:
….. //continua dall’ esempio precedente
laptops.setName("Laptop Computers"); // Modifica
laptopUltraPortable.setName("Ultra-Portable Notebooks"); // Mdifica
laptopTabletPCs.setName("Tablet Computers"); // Modifica
Category laptopBags = new Category("Laptop Bags");
laptops.addChildCategory(laptopBags); // Aggiunge
15
Metodo che crea
l’associazione in tutte
e due i sensi
Queste operazioni in una seconda sessione:
Session session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
// Modifica 3 categorie e ne inserisce una nuova
session.saveOrUpdate(laptops);
tx.commit();
session.close();
Hibernate valuta in autonomia
cosa deve essere updatato e
cosa invece inserito
doriamauro@gmail.com
SaveOrUpdate
saveOrUpdate() ha il seguente effetto:
Tutto questo deve essere reso persistente non importa se nuovo o vecchio
16
E’ una buona idea utilizzare sempre il metodo saveOrUpdate() anche al
posto dei singoli save() o update().
Con saveOrUpdate(), Hibernate determina lo stato dell’oggetto (per
decidere quale operazione fare delle due) in maniera molto più semplice
rispetto ai metodi save() e update() e comunque senza accedere al DB.
Vi è solo un caso (raro) in cui Hibernate effettua una select per
determinare questo: quando l’oggetto utilizza un identificatore composito
con campi normali e non si conosce la sua versione (versioning).
doriamauro@gmail.com
Delete
17
Normalmente la responsabilità della cancellazione degli oggetti ricade
sullo sviluppatore che deve controllare se vi sono references all’oggetto
prima di cancellarlo.
Non è possibile cancellare oggetti a cui sono associati altri oggetti.
Abilitando il delete-orphan, quando si rimuove un oggetto dalla collection
in cui si trova e questi non ha altri references associati, Hibernate
provvedere alla sua rimozione automatica.
E’ una proprietà molto interessante che evita allo sviluppatore di
compiere la doppia azione di eliminare l’oggetto dalla collezione e dallo
stato di persistenza
E’ configurabile soltanto all’interno dei tag ti tipo collections
doriamauro@gmail.com
Delete-orphan
18
Con delete-orphan disattivato:
Item anItem = ... // Caricato in una sessione precedente
Necessarie in quanto la
anItem.getBids().remove(aBid);
semplice rimozione dalla
anItem.getBids().remove(anotherBid);
Session session = sessionFactory.openSession();
collection non è una
Transaction tx = session.beginTransaction();
operazione di persistenza.
session.delete(aBid);
session.delete(anotherBid);
session.saveOrUpdate(anItem);
tx.commit();
session.close();
Item anItem = ... // Caricata in una precedente sessione
anItem.getBids().remove(aBid);
Con delete-orphan attivato:
anItem.getBids().remove(anotherBid);
Session session = sessionFactory.openSession();
Update dell’oggetto prima di
Transaction tx = session.beginTransaction();
chiudere la sessione perché
session.saveOrUpdate(anItem);
anItem è detached; se fosse
tx.commit();
persistent, l’update sarebbe inutile
session.close();
doriamauro@gmail.com
bulk and batch operations
19
Quando si manipola una grande mole di dati, il mondo dei DB
mette a disposizione la possibilità di scrivere delle funzioni
direttamente dulla basedati dette stored procedure
Vedremo come Hibernate mette a disposizione degli strumenti
per gestire le stored procedure e in generale il problema della
massa dei dati.
Hibernate consente di modificare gli oggetti direttamente nel DB
attraverso il liguaggio HQL
doriamauro@gmail.com
Updatare oggetti nel DB
In Hibernate è possibile updatare gli oggetti direttamente nel DB
Queste operazioni sono speciali perché baipassano
completamente il persistence context, quindi non hanno nessun
effetto sugli oggetti in memoria!
Vi sono due modi possibili per procedere:
–
–
20
Operare in un nuovo contesto di persistenza
Utilizzare il metodo refresh()
doriamauro@gmail.com
Updatare oggetti nel DB
Grazie al linguaggio HQL di Hibernate è possibile lanciare le
query che operano direttamente nel DB
Ecco un esempio:
Query q = session.createQuery("update Item i set i.isActive = :isActive");
q.setBoolean("isActive", true);
Numero di oggetti modificati
int updatedItems = q.executeUpdate();
dall’istruzione update
21
La modifica avviene direttamente nel DB.
Le query HQL sono molto simili alle istruzioni SQL ma vanno
specificate le classi al posto delle tabelle e le proprietà al posto
delle colonne.
doriamauro@gmail.com
Updatare oggetti nel DB
Con HQL le update possono essere anche polimorfiche o
modificare la versione degli oggetti (versioning).
Query q = session.createQuery(
"update CreditCard set stolenOn <= :now where type = 'Visa‘" );
q.setTimestamp("now", new Date() );
int updatedCreditCards = q.executeUpdate();
Hibernate conosce la
ripartizione dei dati tra le
tabelle che mappano la
generalizzazione
Query q = session.createQuery("update versioned Item i set i.isActive = :isActive");
q.setBoolean("isActive", true);
int updatedItems = q.executeUpdate();
22
NOTA: le query HQL non si possono fare join, si può coinvolgere una sola
classe alla volta, gli alias sono opzionale e nella were e possibile scrivere
delle sottoquery
doriamauro@gmail.com
Inserire oggetti nel DB
23
Sempre nell’ottica di voler manipolare una grande mole di dati
direttamente nel DB, vediamo un esempio di inserimento.
Si vuole scrivere una procedura che marchi le carte di credito
come rubate e che sposti le informazioni (dati carta, proprietario)
in un’altra tabella)
L’idea è quella di usa una istruzione HQL di insert senza passare
dal persistent context.
Il primo passo è quello di creare una classe apposita da mappare
con il DB e il secondo di scrivere l’istruzione HQL.
doriamauro@gmail.com
Inserire oggetti nel DB
Query q = session.createQuery
("insert into StolenCreditCard (type, number, expMonth, expYear, ownerFirstname,
onwerLastname, ownerLogin, ownerEmailAddress, ownerHomeAddress)
select c.type, c.number, c.expMonth, c.expYear, u.firstname, u.lastname, u.username,
u.email, u.homeAddress
from CreditCard c join c.user u where c.stolenOn is not null“ );
int createdObjects = q.executeUpdate();
24
doriamauro@gmail.com
Batch query
E’ possibile eseguire una query riportando i dati in memoria in gruppi e
non tutti insieme. Ecco un esempio:
Session session = sessionFactory.openSession();
Creazione del cursore
Transaction tx = session.beginTransaction();
ScrollableResults itemCursor = session.createQuery("from Item").scroll();
int count=0;
while ( itemCursor.next() ) {
Spostamento del cursore
Item item = (Item) itemCursor.get(0);
Elemento a partire dalla
modifyItem(item);
posizione corrente
if ( ++count % 100 == 0 ) {
session.flush();
session.clear();
Ogni 100 elementi si
}
ripulisce il persistence
}
context
tx.commit();
session.close();
25
Per avere le migliori prestazioni:
hibernate.jdbc.batch_size = numero
doriamauro@gmail.com
Batch insert
E’ possibile anche inserire e modificare i modalità betch; lo scopo
è quello di fare il flushing dei dati e svuotare il persistent context
durante l’operazione
Session session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
for ( int i=0; i<100000; i++ ) {
Item item = new Item(...);
session.save(item);
if ( i % 100 == 0 ) {
session.flush();
session.clear();
}
}
tx.commit();
session.close();
26
NOTA: sia per le query che
per l’insert, è opportuno
disabilitare la cache di
secondo livello per le classi
persistenti.
Se si utilizza il generatore di
identificativo “identity”, la
batch insert potrebbe non
essere supportata
doriamauro@gmail.com
Data filtering
In alcuni casi, per motivi di sicurezza, si vuole mettere a
disposizione soltanto una parte dei dati presenti sul DB.
Non sempre è possibile scrivere delle query appropriate che
dinamicamente selezionano i dati;
cosa succede se scrivo unaCategoria.getItems()? Ottengo tutti gli
item di quella categoria senza possibilità di evitarne alcuni
27
Una soluzione potrebbero essere le viste (view) sul DB, ma
queste non sono nello standard di SQL e molti DB non le
supportanto
Hibernate consente di filtrare i dati dinamicamente attraverso il
meccanismo dei filter
doriamauro@gmail.com
Dynamic data filter
Prima di tutto il filtro va dichiarato nel file XML di configurazione e si
assegna un nome univoco (la dichiarazione è globale):
<filter-def name="limitItemsByUserRank">
<filter-param name="currentUserRank" type="int"/>
</filter-def>
Nome del filtro
A questo punto è possibile aggiungere il filtro ad una classe:
<class name="Item" table="ITEM">
Una condizione SQL di
...
filtraggio degli oggetti Item
<filter name="limitItemsByUserRank“
condition=":currentUserRank >= (select u.RANK from USER u
where u.USER_ID = SELLER_ID)"/>
</class>
28
Infine, si abilita il filtro per una sessione e si passano gli eventuali
parametri
Filter filter = session.enableFilter("limitItemsByUserRank");
filter.setParameter("currentUserRank", loggedInUser.getRanking());
doriamauro@gmail.com
Dynamic data filter
E’ consentito un passaggio di parameri per la condizione del
filtro.
Il metodo enableFilter() abilita il filtro e torna un oggeto Filter su
cui è possibile invorare diversi metodi. I più utili sono:
–
–
–
–
29
setParameter(): setta un parametro nella condizione del filtro
getFilterDefinition(): lista dei parametri del filtro
validate(): Hibernate torna una exception se il parametro non è
settato.
setParameterList(): setta una lista di argomenti
doriamauro@gmail.com
Dynamic data filter
A questo punto, tutte le query effettuate sulla sessione filtrata
otterranno dei dati soggetti all’azione del filtro.
List<Item> filteredItems = session.createQuery("from Item").list(); List<Item>
filteredItems = session.createCriteria(Item.class).list();
Solo due metodi non sono soggetti al filtro:
–
–
30
Recupero attraverso l’identificatore
Navigazione tramite il metodo set (es: unaCategoria().getItem())
Per evitare questo ultimo aspetto, si può filtrare una collection
doriamauro@gmail.com
Dynamic data filter
31
I campi di applicazione dei filtri sono molti; per citarne alcuni:
–
Sicurezza: per motivi di sicurezza molto spesso è utile limitare
l’accesso ad una parte dei dati e non a tutto il DB
–
Dati spaziali: si potrebbe limitare l’accesso soltanto a quei dati che
sono relativi ad una certa regione spaziale
–
Dati temporali: si potrebbe mettere a disposizione dati in un certo
intervallo di tempo (ad esempio l’ultima settimana)
doriamauro@gmail.com
Lazy strategy
32
La lazy strategy è la capacità di Hibernate di gestire
efficientemente gli oggetti in memoria rimandando (o ritardando)
le operazioni a quando sono strettamente necessario, in modo
del tutto trasparente all’applicazione.
Hibernate applica di default, la lazy strategy per tutti i suoi oggetti
identity e le collections.
La lazy strategy si basa uno dei pattern fondamentali della GoF:
proxy.
Il comportamento di base attuato da Hibernate per il lazy è:
caricare in memoria soltanto gli oggetti che sono stati richiesti
doriamauro@gmail.com
Lazy loading
Vediamo la logica lazy alla base del recupero di un oggetto dal DB.
Consideriamo il caricamento in base all’ identificatore:
Item item = (Item) session.load(Item.class, new Long(123));
Per l’applicazione, il risultato sarà il caricamento in memoria
dell’item“123”, e l’oggetto è nello stato persistent.
In realtà, Hibernate proxa l’oggetto
l’oggetto proxato
reale e passa all’applicazione
Persistence Context
NO
33
SI
item
Proxy
123
123
xxx
yyy
zzz
doriamauro@gmail.com
Lazy loading
Ogni volta che viene richiesto ad Hibernate il caricamento di un
oggetto, esso controlla se può generare un oggetto proxy a runtime per evitare di accedere al DB.
Il proxy ritarda (lazy) il momento dell’accesso ai dati fino a
quando non viene richiesto l’accesso alle proprietà dell’oggetto.
34
Item item = (Item) session.load(Item.class, new Long(123));
item.getId();
item.getDescription(); // Inizializzazione the proxy
L’identificatore è già
presente nel proxy e quindi
non scatta l’accesso al DB
Ecco un altro esempio:
La descrizione no e quindi
scatta l’accesso al DB
Item item = (Item) session.load(Item.class, new Long(123));
User user = (User) session.load(User.class, new Long(1234));
Bid newBid = new Bid("99.99");
newBid.setItem(item);
newBid.setBidder(user);
session.save(newBid);
C’è solo uno spostamento
di puntatori e non è
necessarie select sul DB
doriamauro@gmail.com
Lazy loading
IL lazy loading è estremamento ottimizzato. Al momento del
caricamento dell’oggetto, si ottiene il proxy quando è possibile! Vediamo
un esempio:
L’oggetto notebook è
costruito e salvato in una
sessione
35
…
Category notebook = new Category(…..);
session2.save(notebook);
tx.commit();
L’oggetto notebook diventa
session2.close();
persistent nella nuova
//creazione di una nuova sessione con persistence context vuoto.
sessione
Session session3 = sessionFactory.openSession();
Transaction tx3 = session3.beginTransaction();
Il persistent
session3.update(notebook); //senza questa istruzione si ha il proxy
context già
//recupero l’oggetto notebook salvato in precedenza
possiede
Category altroNotebook= (Category)session3.load(Category.class, new Long(3));
l’oggetto ma di
System.out.println("CLASS: " + altroNotebook.getClass()); //Category
tipo Category;
System.out.println("CLASS: " + notebook.getClass()); //Category
viene restituito
System.out.println (" IDENTICI: " + altroNotebook == notebook); //true
questo e non un
proxy
doriamauro@gmail.com
Lazy loading
Il metodo get(), al contrario di load() non torna mai un proxy.
Se si vuole la classe reale dell’oggetto proxato si può utilizzare
un metodo statico di Hibernate:
HibernateProxyHelper.getClassWithoutInitializingProxy(obj);
36
Una volta che il proxy è “costretto” ad accedere al DB, questo si
dice inizializzato. In questo caso vengono caricati i dati dal DB
ma sempre in una ottica lazy.
Hibernate, quando carica un oggetto, cercherà di minimizzare
l’accesso al DB e di minimizzare l’occupazione di spazio di
memoria per quell’oggetto
doriamauro@gmail.com
Lazy loading
Quando un oggetto Item, ad esempio, è inizializzato, si presenta
così:
Hibernate costruirà una serie di
proxy, uno per ogni associazione
*-a-uno presenti nella classe Item
Hibernate costruirà una sorta di
segnaposto per tutte le collezioni
presenti nella classe Item (si parla
di segnaposto perché Hibernate
wrappa le collection originali con
sue collections interne).
37
Le proprietà value e i componenti
vengono caricati subito (no lazy)
doriamauro@gmail.com
Lazy loading
Le collections vengono inizializzate quando si comincia a
navigare all’interno o quando si chiamano metodi di supporto
come size() o contains().
Per le collections, Hibernate adotta un extra lazy:
<class name="Item" table="ITEM">
...
<set name="bids"
lazy="extra"
….
38
Con lazy=extra, quando vengono effettuate
chiamate ai metodi di supporto, Hibernate
non inizializza la collection ma, per ottenere
le informazioni, effettua query diretta al DB
doriamauro@gmail.com
Eager strategy
39
Cosa succede se si tenta di navigare un oggetto proxato non
inizializzato o con elementi non inizializzati fuori da una sessione
(l’oggetto detached)?
Si ottiene una LazyInitializationException! Hibernate non è più in
grado di recuperare gli oggetti associati perché il persistence
context è ormai chiuso.
Naturalmente, non sempre il proxy è desiderabile soprattutto per
le collections. Hibernate ha una seconda strategia di caricamento
dei dati: eager strategy.
La eager strategy carica subito l’oggetto in ogni sua parte!
doriamauro@gmail.com
Eager strategy
40
Non si vuole disabilitare la gestione proxy per tutte le classi;
questo sarebbe disastroso per le performance dell’applicazioni;
Con lazy=false, si disabilita la lazy strategy per una certa classe.
Ecco un esempio tra Item e User:
<class name="Item" table="ITEM">
...
<many-to-one name="seller“ lazy="false" .../>
<set name="bids"
lazy="false"
inverse="true">
<key column="ITEM_ID"/>
<one-to-many class="Bid"/>
</set>
...
</class>
doriamauro@gmail.com
Domande?
Lazy load
Eager strategy
Batch insert
reachability
cascade
Data filter
41
Batch query
proxy