sql >> Base de Datos >  >> NoSQL >> Redis

Transacciones Redis

Esta página proporciona ejemplos sobre cómo crear transacciones atómicas de Redis con el cliente C# Redis de ServiceStackRedis Service Stack

Cómo crear operaciones atómicas personalizadas en Redis #

Una de las características principales de Redis es la capacidad de construir operaciones atómicas personalizadas. Esto se logra utilizando las operaciones MULTI/EXEC/DISCARD de Redis.

El cliente C# Redis de ServiceStack facilita el uso de transacciones de Redis al proporcionar una IRedisTransaction fuertemente tipada (para cadenas) e IRedisTypedTransaction (para tipos POCO) API con métodos convenientes que le permiten combinar cualquier operación de IRedisClient dentro de una sola transacción.

La creación de una transacción se realiza llamando a IRedisClient.CreateTransaction() . Desde allí, pone en cola todas las operaciones que desea que formen parte de la transacción utilizando uno de los IRedisTransaction.QueueCommand() sobrecargas Después de eso, puede ejecutar todas las operaciones llamando a IRedisTransaction.Commit() que enviará el comando 'EXEC' al servidor Redis ejecutando todos los comandos en cola y procesando sus devoluciones de llamada.

Si no llama al Commit() antes del final del bloque de uso, Dispose() El método invocará automáticamente Rollback() que enviará el comando 'DESCARTAR' eliminando la transacción actual y restableciendo la conexión del cliente de Redis a su estado anterior.

Ejemplos de transacciones de Redis #

A continuación se muestra un ejemplo simple que muestra cómo poner en cola las operaciones de Redis con y sin devolución de llamada.

int callbackResult;
using (var trans = redis.CreateTransaction())
{
  trans.QueueCommand(r => r.Increment("key"));  
  trans.QueueCommand(r => r.Increment("key"), i => callbackResult = i);  

  trans.Commit();
}
//The value of "key" is incremented twice. The latest value of which is also stored in 'callbackResult'.

Otros ejemplos comunes #

El código fuente completo y otros ejemplos comunes se pueden encontrar en la página de pruebas de transacciones comunes.

[Test]
public void Can_Set_and_Expire_key_in_atomic_transaction()
{
    var oneSec = TimeSpan.FromSeconds(1);

    Assert.That(Redis.GetString("key"), Is.Null);
    using (var trans = Redis.CreateTransaction())                  //Calls 'MULTI'
    {
        trans.QueueCommand(r => r.SetString("key", "a"));      //Queues 'SET key a'
        trans.QueueCommand(r => r.ExpireKeyIn("key", oneSec)); //Queues 'EXPIRE key 1'

        trans.Commit();                                        //Calls 'EXEC'

    }                                                              //Calls 'DISCARD' if 'EXEC' wasn't called

    Assert.That(Redis.GetString("key"), Is.EqualTo("a"));
    Thread.Sleep(TimeSpan.FromSeconds(2));
    Assert.That(Redis.GetString("key"), Is.Null);
}

[Test]
public void Can_Pop_priority_message_from_SortedSet_and_Add_to_workq_in_atomic_transaction()
{
    var messages = new List<string> { "message4", "message3", "message2" };

    Redis.AddToList("workq", "message1");

    var priority = 1;
    messages.ForEach(x => Redis.AddToSortedSet("prioritymsgs", x, priority++));

    var highestPriorityMessage = Redis.PopFromSortedSetItemWithHighestScore("prioritymsgs");

    using (var trans = Redis.CreateTransaction())
    {
        trans.QueueCommand(r => r.RemoveFromSortedSet("prioritymsgs", highestPriorityMessage));
        trans.QueueCommand(r => r.AddToList("workq", highestPriorityMessage));	

        trans.Commit();											
    }

    Assert.That(Redis.GetAllFromList("workq"), 
        Is.EquivalentTo(new List<string> { "message1", "message2" }));
    Assert.That(Redis.GetAllFromSortedSet("prioritymsgs"), 
        Is.EquivalentTo(new List<string> { "message3", "message4" }));
}

Ejemplo todo-en-uno #

Este y otros ejemplos se pueden encontrar en el conjunto de pruebas RedisTransactionTests.cs.

Aquí hay ejemplos todo en uno que combinan diferentes operaciones de Redis dentro de una sola transacción:

[Test]
public void Supports_different_operation_types_in_same_transaction()
{
    var incrementResults = new List<int>();
    var collectionCounts = new List<int>();
    var containsItem = false;

    Assert.That(Redis.GetString(Key), Is.Null);
    using (var trans = Redis.CreateTransaction())
    {
        trans.QueueCommand(r => r.Increment(Key), intResult => incrementResults.Add(intResult));
        trans.QueueCommand(r => r.AddToList(ListKey, "listitem1"));
        trans.QueueCommand(r => r.AddToList(ListKey, "listitem2"));
        trans.QueueCommand(r => r.AddToSet(SetKey, "setitem"));
        trans.QueueCommand(r => r.SetContainsValue(SetKey, "setitem"), b => containsItem = b);
        trans.QueueCommand(r => r.AddToSortedSet(SortedSetKey, "sortedsetitem1"));
        trans.QueueCommand(r => r.AddToSortedSet(SortedSetKey, "sortedsetitem2"));
        trans.QueueCommand(r => r.AddToSortedSet(SortedSetKey, "sortedsetitem3"));
        trans.QueueCommand(r => r.GetListCount(ListKey), intResult => collectionCounts.Add(intResult));
        trans.QueueCommand(r => r.GetSetCount(SetKey), intResult => collectionCounts.Add(intResult));
        trans.QueueCommand(r => r.GetSortedSetCount(SortedSetKey), intResult => collectionCounts.Add(intResult));
        trans.QueueCommand(r => r.Increment(Key), intResult => incrementResults.Add(intResult));

        trans.Commit();
    }

    Assert.That(containsItem, Is.True);
    Assert.That(Redis.GetString(Key), Is.EqualTo("2"));
    Assert.That(incrementResults, Is.EquivalentTo(new List<int> { 1, 2 }));
    Assert.That(collectionCounts, Is.EquivalentTo(new List<int> { 2, 1, 3 }));
    Assert.That(Redis.GetAllFromList(ListKey), Is.EquivalentTo(new List<string> { "listitem1", "listitem2" }));
    Assert.That(Redis.GetAllFromSet(SetKey), Is.EquivalentTo(new List<string> { "setitem" }));
    Assert.That(Redis.GetAllFromSortedSet(SortedSetKey), Is.EquivalentTo(new List<string> { "sortedsetitem1", "sortedsetitem2", "sortedsetitem3" }));
}