Contesa dei dati nelle transazioni

Questa pagina descrive la contesa, la serializzabilità e l'isolamento dei dati transazionali. Per esempi di codice delle transazioni, consulta Transazioni e scritture collettive.

Transazioni e contesa dei dati

Affinché una transazione vada a buon fine, i documenti recuperati dalle relative operazioni di lettura devono rimanere invariati dalle operazioni esterne alla transazione. Se un'altra operazione tenta di modificare uno di questi documenti, entra in uno stato di contesa dei dati con la transazione.

Concorrenza dei dati
Quando due o più operazioni competono per il controllo dello stesso documento. Ad esempio, una transazione potrebbe richiedere che un documento rimanga coerente mentre un'operazione concorrente tenta di aggiornare i valori dei campi del documento.

Firestore risolve la contesa dei dati ritardando o interrompendo una delle operazioni. Le librerie client di Firestore ritentano automaticamente le transazioni che non vanno a buon fine a causa della contesa dei dati. Dopo un numero finito di tentativi, l'operazione di transazione non va a buon fine e restituisce un messaggio di errore:

ABORTED: Too much contention on these documents. Please try again.

Quando si decide quale operazione non riuscire o ritardare, il comportamento dipende dal tipo di biblioteca client.

  • Gli SDK mobile/web utilizzano controlli della concorrenza ottimistica.

  • Le librerie client server utilizzano controlli di concorrenza pessimistici.

Contesa dei dati negli SDK mobile/web

Gli SDK mobile/web (piattaforme Apple, Android, web, C++) utilizzano controlli di concorrenza ottimistica per risolvere la contesa dei dati.

Controlli della concorrenza ottimistici
In base all'assunto che la contesa dei dati non sia probabile o che non sia efficiente mantenere i blocchi del database. Le transazioni ottimistiche non utilizzano i blocchi del database per impedire ad altre operazioni di modificare i dati.

Gli SDK mobile/web utilizzano controlli di concorrenza ottimistici, in quanto possono funzionare in ambienti con latenza elevata e una connessione di rete inaffidabile. Il blocco di documenti in un ambiente ad alta latenza causerebbe troppi errori di contesa dei dati.

Negli SDK mobile/web, una transazione tiene traccia di tutti i documenti che leggi all'interno della transazione. La transazione completa le operazioni di scrittura solo se nessuno di questi documenti è stato modificato durante l'esecuzione della transazione. Se un documento è stato modificato, il gestore della transazione riprova la transazione. Se la transazione non riesce a ottenere un risultato chiaro dopo alcuni tentativi, la transazione non va a buon fine a causa della contesa dei dati.

Contesa dei dati nelle librerie client server

Le librerie client server (C#, Go, Java, Node.js, PHP, Python, Ruby) utilizzano controlli della concorrenza pessimistici per risolvere le contese sui dati.

Controlli della concorrenza pessimistici
In base all'ipotesi che sia probabile la contesa dei dati. Le transazioni pessimistiche utilizzano i blocchi del database per impedire ad altre operazioni di modificare i dati.

Le librerie client server utilizzano controlli di concorrenza pessimistici, poiché assumono una latenza ridotta e una connessione affidabile al database.

Nelle librerie client del server, le transazioni applicano blocchi ai documenti che leggono. Il blocco di una transazione su un documento impedisce ad altre transazioni, alle scritture collettive e alle scritture non transazionali di modificare il documento. Una transazione rilascia i blocchi dei documenti al momento del commit. Inoltre, libera i blocchi se scade il tempo di attesa o se non riesce a completare l'operazione per qualsiasi motivo.

Quando una transazione blocca un documento, le altre operazioni di scrittura devono attendere che la transazione rilasci il blocco. Le transazioni acquisiscono i blocchi in ordine cronologico.

Isolamento serializzabile

La contesa dei dati tra le transazioni è strettamente correlata ai livelli di isolamento del database. Il livello di isolamento di un database descrive l'efficacia con cui il sistema gestisce i conflitti tra operazioni simultanee. Il conflitto deriva dai seguenti requisiti del database:

  • Le transazioni richiedono dati accurati e coerenti.
  • Per utilizzare le risorse in modo efficiente, i database eseguono le operazioni in modo simultaneo.

Nei sistemi con un livello di isolamento basso, un'operazione di lettura all'interno di una transazione potrebbe leggere dati inaccurati da modifiche non committate in un'operazione concorrente.

L'isolamento serializzabile definisce il livello di isolamento più elevato. L'isolamento serializzabile significa che:

  • Puoi assumere che il database esegua le transazioni in serie.
  • Le transazioni non sono interessate dalle modifiche non committate nelle operazioni simultanee.

Questa garanzia deve essere valida anche quando il database esegue più transazioni in parallelo. Il database deve implementare controlli di concorrenza per risolvere i conflitti che potrebbero violare questa garanzia.

Firestore garantisce l'isolamento serializzabile delle transazioni. Le transazioni in Firestore vengono serializzate e isolate in base al momento del commit.

Isolamento serializzabile in base all'ora del commit

Firestore assegna a ogni transazione un'ora di commit che rappresenta un singolo punto nel tempo. Quando Firestore esegue il commit delle modifiche di una transazione nel database, puoi assumere che tutte le letture e le scritture all'interno della transazione vengano eseguite esattamente al momento del commit.

L'esecuzione effettiva di una transazione richiede un certo lasso di tempo. L'esecuzione di una transazione inizia prima del momento del commit e l'esecuzione di più operazioni può sovrapporsi. Firestore garantisce l'isolamento serializzabile e assicura che:

  • Firestore esegue il commit delle transazioni in ordine in base al momento del commit.
  • Firestore isola le transazioni dalle operazioni contemporaneamente con un momento di commit successivo.

In caso di contesa dei dati tra operazioni concorrenti, Firestore utilizza controlli di concorrenza ottimistici e pessimistici per risolvere la contesa.

Isolamento all'interno di una transazione

L'isolamento delle transazioni si applica anche alle operazioni di scrittura all'interno di una transazione. Le query e le letture all'interno di una transazione non vedono i risultati delle scritture precedenti all'interno di quella transazione. Anche se modifichi o elimini un documento all'interno di una transazione, tutte le letture del documento nella transazione restituiscono la versione del documento al momento del commit, prima delle operazioni di scrittura della transazione. Le operazioni di lettura non restituiscono nulla se il documento non esisteva.

Problemi di contesa dei dati

Per ulteriori informazioni sulle contese dei dati e su come risolverle, consulta la pagina relativa alla risoluzione dei problemi.