Is there a way to wake a sleeping thread?

后端 未结 7 1708
伪装坚强ぢ
伪装坚强ぢ 2021-01-04 05:13

Is there a way to wake a sleeping thread in C#? So, have it sleep for either a long time and wake it when you want work processed?

相关标签:
7条回答
  • 2021-01-04 05:31

    If your thread is inside a call to Sleep, then there isn't (usually) a way to wake it up. (The only exception I'm aware of is Java, which allows a sleep to be ended early if some other thread calls thread.interrupt().)

    The pattern that you're talking about seems to call for an event: the thread contains a loop, at the top of which it waits for an event to fire. If the event is currently unset, then the thread "sleeps" until some other thread fires the event. At that point, the sleeping thread wakes up and continues its work, until the next time through the loop when it sleeps to wait for another event.

    0 讨论(0)
  • 2021-01-04 05:33

    Would this thread help? C# has good functionality for thread Event handling. I've done most of my work in Python, but C# seems to have solid libraries for thread blocking.

    0 讨论(0)
  • 2021-01-04 05:35

    The best solution would be to use Task objects with the default TaskFactory. This API (introduced in .NET 4.0) uses a pool of threads with work-stealing queues and all that fancy stuff.

    If .NET 4.0 isn't available, then use the ThreadPool, which has a built-in work queue (which does some pool balancing but not on the same scope as the 4.0 thread pool).

    If you really must do it yourself, then I recommend a BlockingCollection<T>, which is a blocking consumer/producer queue added in .NET 4.0.

    If you really must do it yourself and can't use .NET 4.0, then you can use a ManualResetEvent or AutoResetEvent along with a lock-protected queue of work.

    0 讨论(0)
  • 2021-01-04 05:40

    Based on Ilia's suggestion:

    t1 = new Thread(() =>
        {
         while (keepRunning) {
             try {
                 DoWork();
                 Thread.Sleep(all_night_long);
                 }
             catch (ThreadInterruptedException) { }
             }
        });
    t1.Start();
    

    and...

    public void WakeUp()
    {
       t1.Interrupt();
    }
    
    public void StopRunningImmediately()
    {
       keepRunning = false;
       WakeUp(); //immediately
    }
    

    This solution is crude, as there may be other reasons why the ThreadInterruptedException is thrown.

    0 讨论(0)
  • 2021-01-04 05:41

    There is actually a thread.Interrupt() method in C#. While the accepted answer does describes a good pattern that you probably want in your case, I came to this question looking for Thread.Interrupt so I am putting it here.

    0 讨论(0)
  • 2021-01-04 05:45

    Expanding Wim's answer you can also specify a timeout for the WaitHandle.WaitOne() call. So you can use instead of Thread.Sleep(). CancellationToken struct provides you with one so you can signal your tasks like this:

    string SleepAndWakeUp(string value,CancellationToken ct)
    {
        ct.WaitHandle.WaitOne(60000);
        return value;
    }
    
    void Parent()
    {
         CancellationTokenSource cts = new CancellationTokenSource();
         Task.Run(() => SleepAndWakeUp("Hello World!", cts.Token), cts.Token);
         //Do some other work here
         cts.Cancel(); //Wake up the asynch task
    }
    
    0 讨论(0)
提交回复
热议问题