ManualResetEventSlim Klass

Definition

Representerar en trådsynkroniseringshändelse som, när den signaleras, måste återställas manuellt. Den här klassen är ett enkelt alternativ till 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
Arv
ManualResetEventSlim
Attribut
Implementeringar

Exempel

I följande exempel visas hur du använder en 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

Kommentarer

Du kan använda den här klassen för bättre prestanda än ManualResetEvent när väntetiderna förväntas vara mycket korta och när händelsen inte korsar en processgräns. ManualResetEventSlim använder upptagen spinning under en kort tid medan den väntar på att händelsen ska signaleras. När väntetiderna är korta kan spinning vara mycket billigare än att vänta med hjälp av väntehandtag. Men om händelsen inte signaleras inom en viss tidsperiod, ManualResetEventSlim orterar till en vanlig händelse hantera väntan.

Note

I .NET Core och .NET 5+ är den standardmässiga rotationsväntetiden kort: i storleksordningen 10s mikrosekunder, beroende på plattform och processor. Om du förväntar dig att väntetiderna ska vara mycket längre än så kan du fortfarande använda den här klassen i stället för ManualResetEvent (kanske konfigurerad med mindre eller ingen spin-waiting). Prestandaförmånen skulle dock sannolikt bara vara marginell.

Konstruktorer

Name Description
ManualResetEventSlim()

Initierar en ny instans av ManualResetEventSlim klassen med ett initialt tillstånd av icke-signal.

ManualResetEventSlim(Boolean, Int32)

Initierar en ny instans av ManualResetEventSlim klassen med ett booleskt värde som anger om det ursprungliga tillståndet ska anges till signalerat och ett angivet spinnantal.

ManualResetEventSlim(Boolean)

Initierar en ny instans av ManualResetEventSlim klassen med ett booleskt värde som anger om det ursprungliga tillståndet ska anges till signalerat.

Egenskaper

Name Description
IsSet

Hämtar om händelsen har angetts.

SpinCount

Hämtar det antal spinnväntningar som inträffar innan du återgår till en kernelbaserad vänteåtgärd.

WaitHandle

Hämtar det underliggande WaitHandle objektet för detta ManualResetEventSlim.

Metoder

Name Description
Dispose()

Släpper alla resurser som används av den aktuella instansen ManualResetEventSlim av klassen.

Dispose(Boolean)

Släpper de ohanterade resurser som används av ManualResetEventSlim, och släpper eventuellt de hanterade resurserna.

Equals(Object)

Avgör om det angivna objektet är lika med det aktuella objektet.

(Ärvd från Object)
GetHashCode()

Fungerar som standard-hash-funktion.

(Ärvd från Object)
GetType()

Hämtar den aktuella instansen Type .

(Ärvd från Object)
MemberwiseClone()

Skapar en ytlig kopia av den aktuella Object.

(Ärvd från Object)
Reset()

Anger tillståndet för händelsen till icke-signalerad, vilket gör att trådar blockeras.

Set()

Anger tillståndet för händelsen till signalerad, vilket gör att en eller flera trådar som väntar på händelsen kan fortsätta.

ToString()

Returnerar en sträng som representerar det aktuella objektet.

(Ärvd från Object)
Wait()

Blockerar den aktuella tråden tills den aktuella ManualResetEventSlim har angetts.

Wait(CancellationToken)

Blockerar den aktuella tråden tills strömmen ManualResetEventSlim tar emot en signal, samtidigt som en CancellationToken.

Wait(Int32, CancellationToken)

Blockerar den aktuella tråden tills strömmen ManualResetEventSlim har angetts, med hjälp av ett 32-bitars signerat heltal för att mäta tidsintervallet, samtidigt som en CancellationToken.

Wait(Int32)

Blockerar den aktuella tråden tills den aktuella ManualResetEventSlim har angetts med ett 32-bitars signerat heltal för att mäta tidsintervallet.

Wait(TimeSpan, CancellationToken)

Blockerar den aktuella tråden tills den aktuella ManualResetEventSlim har angetts, med hjälp av en TimeSpan för att mäta tidsintervallet, medan du observerar en CancellationToken.

Wait(TimeSpan)

Blockerar den aktuella tråden tills den aktuella ManualResetEventSlim har angetts, med hjälp av en TimeSpan för att mäta tidsintervallet.

Gäller för

Trådsäkerhet

Alla offentliga och skyddade medlemmar ManualResetEventSlim i är trådsäkra och kan användas samtidigt från flera trådar, med undantag för Dispose, som endast får användas när alla andra åtgärder på ManualResetEventSlim har slutförts, och Återställ, som endast ska användas när inga andra trådar kommer åt händelsen.

Se även