Summary of this chapter:
1: Termination status and non-termination status
2: differences between autoresetevent and manualresetevent
3: waithandle. waitone (), etc.
1: Termination status and non-termination status
First, let's talk about the termination and non-termination states of the thread. Both autoresetevent and manualresetevent constructors have bool variables to indicate the termination and non-termination statuses of the thread. True indicates the termination status, and false indicates the non-termination status. View code snippet 1:
Code snippet 1:
AutoResetEvent _autoResetEvent = new AutoResetEvent(false); private void BT_Temp_Click(object sender, RoutedEventArgs e) { Thread t1 = new Thread(this.Thread1Foo); t1.Start(); Thread.Sleep(3000); _autoResetEvent.Set(); } void Thread1Foo() { _autoResetEvent.WaitOne(); MessageBox.Show("t1 end"); }
The execution result of this code is "T1 end" after three seconds ".
If you set:
Autoresetevent _ autoresetevent = new autoresetevent (false );
Changed:
Autoresetevent _ autoresetevent = new autoresetevent (true );
Then "T1 end" will pop up immediately.
In other words, in the terminated state, _ autoresetevent. waitone () does not block the working thread. (PS: manualresetevent)
2: differences between autoresetevent and manualresetevent
Next, let's take a look at the differences between autoresetevent and manualresetevent. We can see code snippet 2 and code snippet 3:
Code Segment 2:
The effect of running this section of code is that after 3 seconds, either "T1 end" or "T2 end" will pop up, and neither of them will pop up. That is to say, one of the processes will end, and the other process will never end.
Code Segment 3:
ManualResetEvent _menuRestEvent = new ManualResetEvent(false); private void BT_Temp_Click(object sender, RoutedEventArgs e) { Thread t1 = new Thread(this.Thread1Foo); t1.Start(); Thread t2 = new Thread(this.Thread2Foo); t2.Start(); Thread.Sleep(3000); _menuRestEvent.Set(); } void Thread1Foo() { _menuRestEvent.WaitOne(); MessageBox.Show("t1 end"); } void Thread2Foo() { _menuRestEvent.WaitOne(); MessageBox.Show("t2 end"); }
The result of this Code section is that after three seconds, "T1 end" and "T2 end" are displayed. That is to say, both processes are finished.
This feature means that autoresetevent only sends signals to one thread rather than to multiple threads. When we need to synchronize multiple threads, we can only use manualresetevent. The underlying reason is that autoresetevent automatically sets the thread status to false after set (), while manualresetevent changes to true after set, the thread is set to false again only after manual reset. That is why their names are auto and manual. To fully verify this feature of manualresetevent, let's look at code snippet 4.
Code snippet 4:
Manualresetevent _ menurestevent = new manualresetevent (false); Private void bt_temp_click (Object sender, routedeventargs e) {thread T1 = new thread (this. thread1foo); t1.start (); thread t2 = new thread (this. thread2foo); t2.start (); thread. sleep (3000); _ menurestevent. set (); // _ menurestevent. reset ();} void thread1foo () {_ menurestevent. waitone (); MessageBox. show ("T1 Step1 end"); // sleep for 1 s, used to wait for the main thread _ menurestevent. reset (); thread. sleep (1000); _ menurestevent. waitone (); MessageBox. show ("T1 step2 end");} void thread2foo () {_ menurestevent. waitone (); MessageBox. show ("T2 Step1 end"); // sleep for 1 s, used to wait for the main thread _ menurestevent. reset (); thread. sleep (1000); _ menurestevent. waitone (); MessageBox. show ("T2 step2 end ");}
In code snippet 4, we pair/_ menurestevent. reset () is annotated, that is, _ menurestevent. after Set (), the thread status is true, the program runs "T1 Step1 end", "T1 step2 end", "T1 step2 end", and "T2 step2 end" in three seconds.
If we remove the comments of // _ menurestevent. Reset (), we will find that "T1 step2 end" and "T2 step2 end" will never pop up. Unless we perform set () on _ menurestevent again in the main thread ().
3: waithandle. waitone (), etc.
No matter what method is used to start a new thread, we can get the waithandle of this thread. The function of using waitone and other methods is to organize the current thread, until other threads send signals to the current thread.
Let's take a look at an example to illustrate this problem. The example is described as follows:
Simulate the client in the communication. When the client is running, the server sends heartbeat data to the client at intervals. The client starts a thread and checks whether heartbeat data is received every three seconds. If there is no heartbeat data, it tells itself that the connection has been disconnected from the server. The Code is as follows:
Thread t; autoresetevent _ autoresetevent = new autoresetevent (false); Private void button#click (Object sender, eventargs e) {T = new thread (New threadstart (delegate {While (true) {// wait for 3 seconds. If no signal is displayed for 3 seconds, the system returns a signal indicating disconnection. Otherwise, the system returns the updated bool Re = _ autoresetevent. waitone (3000); showre (re) ;}}); T. start (); T. isbackground = true;} void showre (bool re) {label1.begininvoke (New methodinvoker (delegate {If (re) {label1.tex T = datetime. now. tostring () + "keep connected";} else {label1.text = datetime. now. tostring () + "disconnected, need to restart" ;}}) ;}// simulate the server to send heartbeat data. Private void button2_click (Object sender, eventargs e) {_ autoresetevent. Set ();}
From: http://www.cnblogs.com/luminji/archive/2010/09/17/1829360.html
AutoResetEvent _autoResetEvent = new AutoResetEvent(false); private void BT_Temp_Click(object sender, RoutedEventArgs e) { Thread t1 = new Thread(this.Thread1Foo); t1.Start(); Thread t2 = new Thread(this.Thread2Foo); t2.Start(); Thread.Sleep(3000); _autoResetEvent.Set(); } void Thread1Foo() { _autoResetEvent.WaitOne(); MessageBox.Show("t1 end"); } void Thread2Foo() { _autoResetEvent.WaitOne(); MessageBox.Show("t2 end"); }