ManualResetEventSlim Classe

Definição

Representa um evento de sincronização de threads que, quando sinalizado, deve ser reiniciado manualmente. Esta classe é uma alternativa leve a ManualResetEvent.

public ref class ManualResetEventSlim : IDisposable
public class ManualResetEventSlim : IDisposable
[System.Runtime.InteropServices.ComVisible(false)]
public class ManualResetEventSlim : IDisposable
type ManualResetEventSlim = class
    interface IDisposable
[<System.Runtime.InteropServices.ComVisible(false)>]
type ManualResetEventSlim = class
    interface IDisposable
Public Class ManualResetEventSlim
Implements IDisposable
Herança
ManualResetEventSlim
Atributos
Implementações

Exemplos

O exemplo seguinte mostra como usar um ManualResetEventSlim.

using System;
using System.Threading;
using System.Threading.Tasks;
class MRESDemo
{

    static void Main()
    {
        MRES_SetWaitReset();
        MRES_SpinCountWaitHandle();
    }
    // Demonstrates:
    //      ManualResetEventSlim construction
    //      ManualResetEventSlim.Wait()
    //      ManualResetEventSlim.Set()
    //      ManualResetEventSlim.Reset()
    //      ManualResetEventSlim.IsSet
    static void MRES_SetWaitReset()
    {
        ManualResetEventSlim mres1 = new ManualResetEventSlim(false); // initialize as unsignaled
        ManualResetEventSlim mres2 = new ManualResetEventSlim(false); // initialize as unsignaled
        ManualResetEventSlim mres3 = new ManualResetEventSlim(true);  // initialize as signaled

        // Start an asynchronous Task that manipulates mres3 and mres2
        var observer = Task.Factory.StartNew(() =>
        {
            mres1.Wait();
            Console.WriteLine("observer sees signaled mres1!");
            Console.WriteLine("observer resetting mres3...");
            mres3.Reset(); // should switch to unsignaled
            Console.WriteLine("observer signalling mres2");
            mres2.Set();
        });

        Console.WriteLine("main thread: mres3.IsSet = {0} (should be true)", mres3.IsSet);
        Console.WriteLine("main thread signalling mres1");
        mres1.Set(); // This will "kick off" the observer Task
        mres2.Wait(); // This won't return until observer Task has finished resetting mres3
        Console.WriteLine("main thread sees signaled mres2!");
        Console.WriteLine("main thread: mres3.IsSet = {0} (should be false)", mres3.IsSet);

        // It's good form to Dispose() a ManualResetEventSlim when you're done with it
        observer.Wait(); // make sure that this has fully completed
        mres1.Dispose();
        mres2.Dispose();
        mres3.Dispose();
    }

    // Demonstrates:
    //      ManualResetEventSlim construction w/ SpinCount
    //      ManualResetEventSlim.WaitHandle
    static void MRES_SpinCountWaitHandle()
    {
        // Construct a ManualResetEventSlim with a SpinCount of 1000
        // Higher spincount => longer time the MRES will spin-wait before taking lock
        ManualResetEventSlim mres1 = new ManualResetEventSlim(false, 1000);
        ManualResetEventSlim mres2 = new ManualResetEventSlim(false, 1000);

        Task bgTask = Task.Factory.StartNew(() =>
        {
            // Just wait a little
            Thread.Sleep(100);

            // Now signal both MRESes
            Console.WriteLine("Task signalling both MRESes");
            mres1.Set();
            mres2.Set();
        });

        // A common use of MRES.WaitHandle is to use MRES as a participant in 
        // WaitHandle.WaitAll/WaitAny.  Note that accessing MRES.WaitHandle will
        // result in the unconditional inflation of the underlying ManualResetEvent.
        WaitHandle.WaitAll(new WaitHandle[] { mres1.WaitHandle, mres2.WaitHandle });
        Console.WriteLine("WaitHandle.WaitAll(mres1.WaitHandle, mres2.WaitHandle) completed.");

        // Clean up
        bgTask.Wait();
        mres1.Dispose();
        mres2.Dispose();
    }
}
Imports System.Threading
Imports System.Threading.Tasks

Module MRESDemo

    Sub Main()

    End Sub
    ' Demonstrates:
    ' ManualResetEventSlim construction
    ' ManualResetEventSlim.Wait()
    ' ManualResetEventSlim.Set()
    ' ManualResetEventSlim.Reset()
    ' ManualResetEventSlim.IsSet
    Private Sub MRES_SetWaitReset()
        ' initialize as unsignaled
        Dim mres1 As New ManualResetEventSlim(False)
        ' initialize as unsignaled
        Dim mres2 As New ManualResetEventSlim(False)
        ' initialize as signaled
        Dim mres3 As New ManualResetEventSlim(True)

        ' Start an asynchronous Task that manipulates mres3 and mres2
        Dim observer = Task.Factory.StartNew(
            Sub()
                mres1.Wait()
                Console.WriteLine("observer sees signaled mres1!")
                Console.WriteLine("observer resetting mres3...")
                mres3.Reset()
                ' should switch to unsignaled
                Console.WriteLine("observer signalling mres2")
                mres2.[Set]()
            End Sub)

        Console.WriteLine("main thread: mres3.IsSet = {0} (should be true)", mres3.IsSet)
        Console.WriteLine("main thread signalling mres1")
        mres1.[Set]()
        ' This will "kick off" the observer Task
        mres2.Wait()
        ' This won't return until observer Task has finished resetting mres3
        Console.WriteLine("main thread sees signaled mres2!")
        Console.WriteLine("main thread: mres3.IsSet = {0} (should be false)", mres3.IsSet)

        ' make sure that observer has fully completed
        observer.Wait()
        ' It's good form to Dispose() a ManualResetEventSlim when you're done with it
        mres1.Dispose()
        mres2.Dispose()
        mres3.Dispose()
    End Sub

    ' Demonstrates:
    ' ManualResetEventSlim construction w/ SpinCount
    ' ManualResetEventSlim.WaitHandle
    Private Sub MRES_SpinCountWaitHandle()
        ' Construct a ManualResetEventSlim with a SpinCount of 1000
        ' Higher spincount => longer time the MRES will spin-wait before taking lock
        Dim mres1 As New ManualResetEventSlim(False, 1000)
        Dim mres2 As New ManualResetEventSlim(False, 1000)

        Dim bgTask As Task = Task.Factory.StartNew(
            Sub()
                ' Just wait a little
                Thread.Sleep(100)

                ' Now signal both MRESes
                Console.WriteLine("Task signalling both MRESes")
                mres1.[Set]()
                mres2.[Set]()
            End Sub)

        ' A common use of MRES.WaitHandle is to use MRES as a participant in 
        ' WaitHandle.WaitAll/WaitAny. Note that accessing MRES.WaitHandle will
        ' result in the unconditional inflation of the underlying ManualResetEvent.
        WaitHandle.WaitAll(New WaitHandle() {mres1.WaitHandle, mres2.WaitHandle})
        Console.WriteLine("WaitHandle.WaitAll(mres1.WaitHandle, mres2.WaitHandle) completed.")

        ' Wait for bgTask to complete and clean up
        bgTask.Wait()
        mres1.Dispose()
        mres2.Dispose()
    End Sub
End Module

Observações

Pode usar esta classe para melhor desempenho do que ManualResetEvent quando se espera que os tempos de espera sejam muito curtos e quando o evento não ultrapassa o limite do processo. ManualResetEventSlim Usa a fiação ocupada por um curto período enquanto espera que o evento seja sinalizado. Quando os tempos de espera são curtos, girar pode ser muito menos dispendioso do que esperar usando alças de espera. No entanto, se o evento não for sinalizado dentro de um determinado período de tempo, ManualResetEventSlim recorre a uma espera regular para o evento.

Note

Em .NET Core e .NET 5+, a duração padrão de espera da rotação é curta: na ordem de dezenas de microssegundos, dependendo da plataforma e do processador. Se esperares que os tempos de espera sejam muito mais longos, ainda podes usar esta classe em vez de ManualResetEvent (talvez configurado com menos ou nenhuma spin-waiting). No entanto, o benefício de desempenho seria provavelmente apenas marginal.

Construtores

Name Description
ManualResetEventSlim()

Inicializa uma nova instância da ManualResetEventSlim classe com um estado inicial de não sinalizado.

ManualResetEventSlim(Boolean, Int32)

Inicializa uma nova instância da ManualResetEventSlim classe com um valor booleano indicando se deve definir o estado inicial para sinalizado e uma contagem de spins especificada.

ManualResetEventSlim(Boolean)

Inicializa uma nova instância da ManualResetEventSlim classe com um valor booleano que indica se deve definir o estado inicial para sinalizado.

Propriedades

Name Description
IsSet

Percebe se o evento está marcado.

SpinCount

Obtém o número de spin waits que ocorrerão antes de recorrer a uma operação de espera baseada no kernel.

WaitHandle

Obtém o objeto subjacente WaitHandle para este ManualResetEventSlim.

Métodos

Name Description
Dispose()

Liberta todos os recursos usados pela instância atual da ManualResetEventSlim classe.

Dispose(Boolean)

Liberta os recursos não geridos usados pelo ManualResetEventSlim, e opcionalmente liberta os recursos geridos.

Equals(Object)

Determina se o objeto especificado é igual ao objeto atual.

(Herdado de Object)
GetHashCode()

Serve como função de hash predefinida.

(Herdado de Object)
GetType()

Obtém o Type da instância atual.

(Herdado de Object)
MemberwiseClone()

Cria uma cópia superficial do atual Object.

(Herdado de Object)
Reset()

Define o estado do evento para não sinalizado, o que faz com que as threads bloqueem.

Set()

Define o estado do evento para sinalizado, o que permite que um ou mais threads aguardem que o evento prossiga.

ToString()

Devolve uma cadeia que representa o objeto atual.

(Herdado de Object)
Wait()

Bloqueia o fio atual até que a corrente ManualResetEventSlim esteja definida.

Wait(CancellationToken)

Bloqueia a linha de corrente até que esta ManualResetEventSlim receba um sinal, enquanto observa um CancellationToken.

Wait(Int32, CancellationToken)

Bloqueia o fio atual até que a corrente ManualResetEventSlim seja definida, usando um inteiro assinado de 32 bits para medir o intervalo de tempo, enquanto observa um CancellationToken.

Wait(Int32)

Bloqueia a thread corrente até que a corrente ManualResetEventSlim seja definida, usando um inteiro assinado de 32 bits para medir o intervalo de tempo.

Wait(TimeSpan, CancellationToken)

Bloqueia o fio de corrente até que a corrente ManualResetEventSlim seja definida, usando a TimeSpan para medir o intervalo de tempo, enquanto observa um CancellationToken.

Wait(TimeSpan)

Bloqueia a linha de corrente até que a corrente ManualResetEventSlim seja definida, usando a TimeSpan para medir o intervalo de tempo.

Aplica-se a

Segurança de Thread

Todos os membros públicos e protegidos de ManualResetEventSlim são seguros para threads e podem ser usados simultaneamente a partir de múltiplas threads, com exceção do Dispose, que só deve ser usado quando todas as outras operações no ManualResetEventSlim estão concluídas, e do Reset, que só deve ser usado quando nenhuma outra thread está a aceder ao evento.

Ver também