Transazioni e modalità di blocco delle raccolte Reliable Collections in Azure Service Fabric
Transazione
Una transazione è una sequenza di operazioni eseguite in un'unica unità logica di lavoro. Mostra le proprietà ACID comuni (atomicità, coerenza, isolamento, durabilità) delle transazioni di database:
- Atomicità: una transazione deve essere un'unità di lavoro atomica. In altre parole, devono essere eseguite tutte le modifiche dei dati oppure non ne deve essere eseguita nessuna.
- Coerenza: al termine della transazione, tutti i dati devono essere coerenti e tutte le strutture di dati interne devono risultare corrette.
- Isolamento: le modifiche eseguite da transazioni simultanee devono essere isolate da quelle eseguite da qualsiasi altra transazione simultanea. Il livello di isolamento usato per un'operazione all'interno di un oggetto ITransaction è determinato dall'interfaccia IReliableState che esegue l'operazione.
- Durabilità: gli effetti di una transazione completata sono permanenti all'interno del sistema. Le modifiche eseguite rimangono valide anche in caso di errore del sistema.
Livelli di isolamento
Il livello di isolamento definisce il grado in cui la transazione deve essere isolata dalle modifiche apportate da altre transazioni. Le raccolte Reliable Collections supportano due livelli di isolamento:
- Repeatable Read: specifica che le istruzioni non possono leggere dati modificati da altre transazioni di cui non è ancora stato eseguito il commit e che nessun'altra transazione può modificare i dati letti dalla transazione corrente, finché quest'ultima non viene completata.
- Snapshot: specifica che i dati letti da qualsiasi istruzione in una transazione rappresentano la versione coerente dal punto di vista transazionale dei dati esistenti al momento dell'avvio della transazione. La transazione può quindi riconoscere solo le modifiche dei dati di cui è stato eseguito il commit prima dell'avvio della transazione. Le modifiche ai dati apportate da altre transazioni dopo l'avvio della transazione corrente non sono visibili per le istruzioni eseguite nella transazione corrente. È come se le istruzioni di una transazione ottenessero uno snapshot dei dati di cui è stato eseguito il commit così come si presentavano al momento dell'avvio della transazione. Gli snapshot tra le Reliable Collections sono coerenti.
Le raccolte Reliable Collections scelgono automaticamente il livello di isolamento da usare per una determinata operazione di lettura a seconda dell'operazione stessa e del ruolo della replica al momento della creazione della transazione. La tabella seguente descrive i valori predefiniti del livello di isolamento per le operazioni Reliable Dictionary e Reliable Queue.
Operazione\Ruolo | Primaria | Secondari |
---|---|---|
Lettura di entità singola | Repeatable Read | Snapshot |
Enumerazione, conteggio | Snapshot | Snapshot |
Nota
Alcuni esempi comuni per le operazioni sulla singola entità sono IReliableDictionary.TryGetValueAsync
e IReliableQueue.TryPeekAsync
.
Gli oggetti ReliableDictionary e ReliableQueue supportano entrambi il criterio Read Your Writes. In altri termini, qualsiasi operazione di scrittura all'interno di una transazione sarà visibile a una lettura successiva appartenente alla stessa transazione.
Locks
Nelle raccolte Reliable Collections tutte le transazioni implementano un rigoroso blocco in due fasi: una transazione non rilascia i blocchi acquisiti fino a quando non termina con un'interruzione o un commit.
L'operazione Reliable Dictionary usa il blocco a livello di riga per tutte le operazioni sulla singola entità.
L'operazione Reliable Queue bilancia la concorrenza in cambio di una proprietà FIFO transazionale rigorosa.
Reliable Queue usa i blocchi a livello di operazione che consentono una transazione con TryPeekAsync
e/o TryDequeueAsync
e una transazione con EnqueueAsync
per volta.
Si noti che per mantenere il modello FIFO, se TryPeekAsync
o TryDequeueAsync
rileva che la coda affidabile è vuota, bloccherà anche EnqueueAsync
.
Le operazioni di scrittura acquisiscono sempre blocchi esclusivi. Per le operazioni di lettura, il blocco dipende da due fattori:
- Le operazioni di lettura eseguite con Shapshot Isolation sono prive di blocchi.
- Per impostazione predefinita, ogni operazione di lettura ripetibile acquisisce blocchi condivisi.
- Tuttavia, per ogni operazione di lettura che supporta Repeatable Read, l'utente può chiedere un blocco di aggiornamento al posto del blocco condiviso. Il blocco di aggiornamento è asimmetrico e viene usato per impedire una forma comune di deadlock che si verifica quando più transazioni bloccano le risorse per un potenziale aggiornamento in un secondo momento.
La matrice di compatibilità dei blocchi è disponibile nella tabella seguente:
Richiesto\Concesso | None | Condiviso | Aggiornamento | Esclusivo |
---|---|---|---|---|
Condiviso | Nessun conflitto | Nessun conflitto | Conflitto | Conflitto |
Aggiornamento | Nessun conflitto | Nessun conflitto | Conflitto | Conflitto |
Esclusivo | Nessun conflitto | Conflitto | Conflitto | Conflitto |
Per il rilevamento dei deadlock viene usato un argomento timeout delle API raccolte affidabili. Si supponga, ad esempio, che due transazioni (T1 e T2) stiano cercando di leggere e aggiornare K1. È possibile che si verifichi un deadlock, poiché entrambe le transazioni si concludono con il blocco condiviso. In questo caso, potrebbe verificarsi il timeout di una o di entrambe le operazioni. In questo scenario, un blocco di aggiornamento potrebbe impedire un deadlock di questo tipo.
Passaggi successivi
- Lavorare con le raccolte Reliable Collections
- Notifiche di Reliable Services
- Eseguire il backup e il ripristino di Reliable Services (ripristino di emergenza)
- Reliable State Manager configuration (Configurazione di Reliable State Manager)
- Guida di riferimento per gli sviluppatori per Reliable Collections