有时间我们在使用多线程的时候,需要取消线程的执行,可以使用CancellationTokenSource来取消对Task开辟多线程的取消
如下:我们点击一个按钮开启线程,然后点击取消按钮取消该线程的执行
CancellationTokenSource cts ; /// <summary> /// 开启线程 /// </summary> private void button1_Click(object sender, EventArgs e) { //重新实例化cts.Token就会不一样,不然再次点击的时候会报错,提示线程已经执行完毕 cts = new CancellationTokenSource(); Task task = new Task(() => { Thread.Sleep(1500); try { cts.Token.ThrowIfCancellationRequested(); MessageBox.Show("线程被执行"); } catch (Exception ex) { MessageBox.Show("线程被取消"); } }, cts.Token); task.Start(); //如果放到这里有可能线程还没有开始执行就被取消了 //cts.Cancel(true); } /// <summary> /// 取消线程 /// </summary> private void button2_Click(object sender, EventArgs e) { cts.Cancel(); }
几点关键问题解释:
1:为什么线程里边去调用 Thread.Sleep(1500);
因为如果不让线程暂停一下,你在点击开启线程后。还没有来得及点击取消线程,线程都已经执行完毕了这个时候
再去取消线程是没有什么意义的。
2: cts.Cancel()
这句话其实只是设置一个状态,设置该对象一个熟悉为true而已,也就是设置cts.IsCancellationRequested为ture。
所以他根本没有真正的去取消线程的执行,只是设置一个状态而已。
3:cts.Token.ThrowIfCancellationRequested();
因为上面说的cts.Cancel()只是设置一个状态而已,线程不会真正被取消。其实线程开启后,现在都是要执行完的,
我们并不能真正去取消线程的执行,只是从逻辑上去取消线程需要处理的逻辑。
这句话的意思其实就是检查cts.Cancel()设置的属性状态,也就是cts.IsCancellationRequested是否为true,如果为
true就抛出一个异常,从而终止线程的执行
也就是说其实这句话完全可以我们自己来实现
同时取消多个线程:
同时取消多个线程,其实也很简单,我们不需要有多少个线程就去实例化多少个CancellationTokenSource,
多个线程绑定顶一个CancellationTokenSource的token然后调用该对象的cancel就可以同时取消多个线程了。
CancellationTokenSource cts; /// <summary> /// 开启线程 /// </summary> private void button1_Click(object sender, EventArgs e) { //重新实例化cts.Token就会不一样,不然再次点击的时候会报错,提示线程已经执行完毕 cts = new CancellationTokenSource(); for (int i = 0; i < 5; i++) { Task task = new Task((obj) => { Thread.Sleep(1500); try { //cts.Token.ThrowIfCancellationRequested(); //检查状态,检查线程是否已经被取消,如果取消则自己跑出来一个线程 if (cts.Token.IsCancellationRequested) throw new OperationCanceledException(); MessageBox.Show("线程" + obj + "被执行"); } catch (Exception ex) { MessageBox.Show("线程" + obj + "被取消"); } }, i, cts.Token); task.Start(); } //如果放到这里有可能线程还没有开始执行就被取消了 //cts.Cancel(true); } /// <summary> /// 取消线程 /// </summary> private void button2_Click(object sender, EventArgs e) { cts.Cancel(); }
欢迎加群讨论技术,群:677373950(满了,可以加,但通过不了),2群:656732739