Confrontare DML e mutazioni

Il linguaggio di manipolazione dei dati (DML) e le mutazioni sono due API in Spanner che puoi utilizzare per modificare i dati. Ciascuno di essi offre funzionalità di manipolazione dei dati simili. Questa pagina confronta entrambi gli approcci.

Che cos'è il Data Manipulation Language (DML)?

Il linguaggio di manipolazione dei dati (DML) in Spanner ti consente di manipolare i dati nelle tabelle del database utilizzando le istruzioni INSERT, UPDATE e DELETE. Puoi eseguire istruzioni DML utilizzando le librerie client, la console Google Cloud e gcloud spanner.

Spanner offre le due seguenti implementazioni dell'esecuzione DML, ognuna con proprietà diverse.

  • DML standard: adatto all'elaborazione delle transazioni online (OLTP) standard. carichi di lavoro con scale out impegnativi.

    Per maggiori informazioni, inclusi esempi di codice, consulta Utilizzo di DML

  • DML partizionato: progettato per aggiornamenti ed eliminazioni collettivi, come negli esempi riportati di seguito.

    • Pulizia e garbage collection periodici. Alcuni esempi sono l'eliminazione di righe vecchie o l'impostazione di colonne su NULL.

    • Eseguire il backfill delle nuove colonne con i valori predefiniti. Un esempio è l'utilizzo di un'istruzione UPDATE per impostare il valore di una nuova colonna su False se è NULL.

    Per ulteriori informazioni, inclusi esempi di codice, consulta Utilizzare il DML partizionato.

    Puoi utilizzare le scritture in batch per un numero elevato di operazioni di scrittura senza operazioni di lettura che non richiedono transazioni atomiche. Per ulteriori informazioni, consulta la sezione Modificare i dati utilizzando le scritture collettive.

Che cosa sono le mutazioni?

Una mutazione rappresenta una sequenza di inserimenti, aggiornamenti ed eliminazioni che Spanner applica in modo atomico a righe e tabelle diverse in un database. Puoi includere operazioni che si applicano a righe o tabelle diverse in una mutazione. Dopo aver definito una o più mutazioni contenenti una o più scritture, devi applicare la mutazione per eseguire il commit delle scritture. Ogni modifica viene applicata nell'ordine in cui è stata aggiunta alla mutazione.

Per ulteriori informazioni, inclusi esempi di codice, consulta Inserire, aggiornare ed eliminare i dati utilizzando le mutazioni.

Confronto delle caratteristiche tra DML e mutazioni

La tabella seguente riassume il supporto di DML e mutazioni del database comune il funzionamento e le caratteristiche del prodotto.

Operazioni DML Mutazioni
Inserisci dati Supportato Supportato
Elimina dati Supportato Supportato
Aggiorna dati Supportato Supportato
Inserisci o ignora dati Supportato Non supportato
Lettura delle scritture (RYW) Supportato Non supportato
Inserire o aggiornare i dati (upsert) Supportato Supportato
Sintassi SQL Supportato Non supportato
Controllo dei vincoli Dopo ogni istruzione Al momento del commit

Il supporto di DML e mutazioni divergono per le seguenti funzionalità:

  • Leggi le tue scritture: lettura dei risultati non impegnati in un'istanza transazione. Le modifiche apportate utilizzando le istruzioni DML sono visibili alle istruzioni successive nella stessa transazione. Questo è diverso dall'utilizzo delle mutazioni, in cui le modifiche non sono visibili in nessuna lettura (incluse le letture eseguite nella stessa transazione) fino all'esecuzione dell'commit della transazione. Questo perché le mutazioni in una transazione vengono messe in coda lato client (localmente) e inviate al server nell'ambito dell'operazione di commit. Di conseguenza, le mutazioni nella richiesta di commit non sono visibili alle istruzioni SQL o DML all'interno della stessa transazione.

  • Controllo dei vincoli: Spanner controlla i vincoli dopo ogni istruzione DML. Questo è diverso dall'utilizzo delle mutazioni, in cui Spanner immagazzina le mutazioni nel client fino al commit e controlla le limitazioni al momento del commit. La valutazione dei vincoli dopo ogni istruzione DML consente a Spanner per garantire che i dati restituiti da una query successiva nello stesso la transazione restituisce dati coerenti con lo schema.

  • Sintassi SQL: la DML fornisce un modo convenzionale per manipolare i dati. Puoi riutilizzare le competenze SQL per modificare i dati utilizzando l'API DML.

Best practice: evita di combinare DML e mutazione nella stessa transazione

Se una transazione contiene sia istruzioni DML sia mutazioni nel commit Spanner esegue le istruzioni DML prima delle mutazioni. Per evitare di dover tenere conto dell'ordine di esecuzione nel codice della libreria client, devi utilizzare le istruzioni DML o le mutazioni in una singola transazione, ma non entrambe.

Il seguente esempio Java illustra un comportamento potenzialmente sorprendente. Il codice inserisce due righe negli album utilizzando l'API Mutation. Lo snippet, quindi, chiama executeUpdate() per aggiornare le righe e le chiamate appena inserite executeQuery() per leggere gli album aggiornati.

static void updateMarketingBudget(DatabaseClient dbClient) {
  dbClient
      .readWriteTransaction()
      .run(
          new TransactionCallable<Void>() {
            @Override
            public Void run(TransactionContext transaction) throws Exception {
               transaction.buffer(
                    Mutation.newInsertBuilder("Albums")
                        .set("SingerId")
                        .to(1)
                        .set("AlbumId")
                        .to(1)
                        .set("AlbumTitle")
                        .to("Total Junk")
                        .set("MarketingBudget")
                        .to(800)
                        .build());
               transaction.buffer(
                    Mutation.newInsertBuilder("Albums")
                        .set("SingerId")
                        .to(1)
                        .set("AlbumId")
                        .to(2)
                        .set("AlbumTitle")
                        .to("Go Go Go")
                        .set("MarketingBudget")
                        .to(200)
                        .build());

                // This UPDATE will not include the Albums inserted above.
                String sql =
                  "UPDATE Albums SET MarketingBudget = MarketingBudget * 2"
                      + " WHERE SingerId = 1";
                long rowCount = transaction.executeUpdate(Statement.of(sql));
                System.out.printf("%d records updated.\n", rowCount);

                // Read a newly updated record.
                sql =
                  "SELECT SingerId, AlbumId, AlbumTitle FROM Albums"
                      + " WHERE SingerId = 1 AND MarketingBudget < 1000";
                ResultSet resultSet =
                                 transaction.executeQuery(Statement.of(sql));
                while (resultSet.next()) {
                   System.out.printf(
                        "%s %s\n",
                        resultSet.getString("FirstName"),
                        resultSet.getString("LastName"));
                }
                return null;
              }
            });
}

Se eseguissi questo codice, vedrai 0 record aggiornati. Perché? Questo accade perché le modifiche apportate utilizzando le mutazioni non sono visibili alle istruzioni successive fino all'esecuzione dell'commit della transazione. Idealmente, dovremmo avere scritture memorizzate nel buffer solo alla fine della transazione.

Passaggi successivi