ManualResetEventSlim Clase
Definición
Importante
Parte de la información hace referencia a la versión preliminar del producto, que puede haberse modificado sustancialmente antes de lanzar la versión definitiva. Microsoft no otorga ninguna garantía, explícita o implícita, con respecto a la información proporcionada aquí.
Representa un evento de sincronización de subprocesos que, cuando se señale, se debe restablecer manualmente. Esta clase es una alternativa ligera 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
- Herencia
-
ManualResetEventSlim
- Atributos
- Implementaciones
Ejemplos
En el ejemplo siguiente se muestra cómo usar .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
Comentarios
Puede usar esta clase para mejorar el rendimiento que ManualResetEvent cuando se espera que los tiempos de espera sean muy cortos y cuando el evento no cruza un límite de proceso. ManualResetEventSlim usa giros de ocupado durante un breve tiempo mientras se espera que se señale el evento. Cuando los tiempos de espera son breves, los giros pueden ser mucho más económicos que si se espera con identificadores de espera. Sin embargo, si no se señala el evento dentro de cierto período de tiempo, ManualResetEventSlim recurre a una espera de controlador de evento habitual.
Nota
En .NET Core y .NET 5+, la duración predeterminada de la espera de número es corta: en el orden de 10s de microsegundos, según la plataforma y el procesador. Si espera que los tiempos de espera sean mucho más largos que eso, puede seguir usando esta clase en lugar de ManualResetEvent (quizás configurada con menos o sin espera de número). Sin embargo, es probable que la ventaja de rendimiento sea solo marginal.
Constructores
ManualResetEventSlim() |
Inicializa una nueva instancia de la clase ManualResetEventSlim con el estado inicial establecido en nonsignaled. |
ManualResetEventSlim(Boolean) |
Inicializa una nueva instancia de la clase ManualResetEventSlim con un valor booleano que indica si hay que establecer el estado inicial en señalado. |
ManualResetEventSlim(Boolean, Int32) |
Inicializa una instancia nueva de la clase ManualResetEventSlim con un valor booleano que indica si hay que establecer el estado inicial en señalado y un recuento circular especificado. |
Propiedades
IsSet |
Obtiene un valor que indica si se ha establecido el evento. |
SpinCount |
Obtiene el número de esperas circulares que se van a producir antes de una operación de espera basada en kernel. |
WaitHandle |
Obtiene el objeto WaitHandle subyacente de este objeto ManualResetEventSlim. |
Métodos
Dispose() |
Libera todos los recursos usados por la instancia actual de la clase ManualResetEventSlim. |
Dispose(Boolean) |
Libera los recursos no administrados utilizados por el objeto ManualResetEventSlim y, de forma opcional, libera los recursos administrados. |
Equals(Object) |
Determina si el objeto especificado es igual que el objeto actual. (Heredado de Object) |
GetHashCode() |
Sirve como la función hash predeterminada. (Heredado de Object) |
GetType() |
Obtiene el Type de la instancia actual. (Heredado de Object) |
MemberwiseClone() |
Crea una copia superficial del Object actual. (Heredado de Object) |
Reset() |
Establece el estado del evento en no señalado, por lo que se bloquean los subprocesos. |
Set() |
Establece el estado del evento en señalado, lo que permite la continuación de uno o varios subprocesos que están esperando en el evento. |
ToString() |
Devuelve una cadena que representa el objeto actual. (Heredado de Object) |
Wait() |
Bloquea el subproceso actual hasta que se establezca el objeto ManualResetEventSlim actual. |
Wait(CancellationToken) |
Bloquea el subproceso actual hasta que el objeto ManualResetEventSlim actual reciba una señal, mientras se observa un token CancellationToken. |
Wait(Int32) |
Bloquea el subproceso actual hasta que se establezca el objeto ManualResetEventSlim actual, usando un entero de 32 bits con signo para medir el intervalo de tiempo. |
Wait(Int32, CancellationToken) |
Bloquea el subproceso actual hasta que se establezca el objeto ManualResetEventSlim actual, usando un entero de 32 bits con signo para medir el intervalo de tiempo, mientras se observa un token CancellationToken. |
Wait(TimeSpan) |
Bloquea el subproceso actual hasta que se establezca el objeto ManualResetEventSlim actual, utilizando un objeto TimeSpan para medir el intervalo de tiempo. |
Wait(TimeSpan, CancellationToken) |
Bloquea el subproceso actual hasta que se establezca el objeto ManualResetEventSlim actual, usando un objeto TimeSpan para medir el intervalo de tiempo, mientras se observa un token CancellationToken. |
Se aplica a
Seguridad para subprocesos
Todos los miembros públicos y protegidos de ManualResetEventSlim son seguros para subprocesos y se pueden usar simultáneamente desde varios subprocesos, con la excepción de Dispose, que solo se debe usar cuando todas las demás operaciones de han ManualResetEventSlim finalizado y Restablecer, que solo se deben usar cuando ningún otro subproceso acceda al evento.