Why
- 主要是公司需要实现的需求中有关于这种的情况,需要在后台操作的时候,显示一个页面阻塞
UI
主线程,然后后台继续跑算法模块。因为项目比较老,所以这一块估计经手了一些人导致内部冗余不堪,画了一两天时候看了代码我这边重新整合了一部分,同时之前未曾实现的部分也补上了。 - 主要使用的是
Dev
的SplashScreenManager
管理,省去了一部分工作,公司代码不方便展示,所以我目前这边自己剥离了一份出来。
Code
等待窗体
- 那么对于两种窗体分别创建子,第一种必须是继承
SplashScreen
,以下是设计代码
public WaitScreenForm()
{
InitializeComponent();
}
private void simpleButton1_Click(object sender, EventArgs e)
{
//这个就是中途取消的按钮
SplashScreenManager.CloseForm();
SplashScreenManager.Default?.CloseWaitForm();
//这个助手类后续会说到 表示调用所有窗体取消的事件
WaitHelper.Cancel();
//这个助手类后续会说到 表示重置初始状态
WaitHelper.Reset();
}
public override void ProcessCommand(Enum cmd, object arg)
{
//用于实时展示状态属性在变化 后续可以自行添加代码逻辑
this.groupControl1.Text = DateTime.Now.ToString();
base.ProcessCommand(cmd, arg);
}
- 以下是窗体设计的界面
- 第二种相对比较简单,只需要继承自
WaitForm
即可
public partial class MyWaitForm : WaitForm
{
public MyWaitForm()
{
InitializeComponent();
}
}
- 以下是窗体设计
助手类设计
- 首先设计的助手类是静态类,内部有一个开始等待的函数,传入一个需要锁定的主窗体和是否是等待窗体。
public static class WaitHelper
{
//在是第一种窗体的时候绑定的取消事件
public static event Action OnWaitEventHandler;
//在等待时需要锁定的父窗体
public static Form parentForm = null;
//开始等待
public static void BeginWait(Form _parentForm, bool isWait = true)
{
parentForm = _parentForm;
parentForm.Enabled = false;
if (isWait)
{
SplashScreenManager.ShowForm(_parentForm, typeof(MyWaitForm), true, true, true, ParentFormState.Locked);
}
else
{
SplashScreenManager.ShowForm(_parentForm, typeof(WaitScreenForm), true, true, true, ParentFormState.Locked);
}
}
//取消
public static void Cancel()
{
OnWaitEventHandler?.Invoke();
}
//设置默认状态
public static void Reset()
{
OnWaitEventHandler = null;
//此处必须调用invoke
parentForm.Invoke(new Action(() =>
{
parentForm.Enabled = true;
parentForm = null;
}));
}
}
主窗体
- 主窗体的测试代码,按照需求一定是需要使用线程的,我测试过
thread
,task
和BackgroundWorker
,后面两个其实都无法即时的取消线程,不管是task
配合CancellationTokenSource
还是BackgroundWorker
的取消,都需要在循环内部使用一定的判断,因为其实并不是很推荐在运算时直接终端线程,这样会导致数据问题和未知的错误。
//private CancellationTokenSource tokenSource = new CancellationTokenSource();
private Thread thread = null;
public MainForm()
{
InitializeComponent();
}
private void simpleButton1_Click(object sender, EventArgs e)
{
SplashScreenManager.ShowForm(this, typeof(WaitScreenForm), true, true, true, ParentFormState.Locked);
}
private void simpleButton2_Click(object sender, EventArgs e)
{
SplashScreenManager.CloseForm(true, 0, this);
}
private void simpleButton3_Click(object sender, EventArgs e)
{
//tokenSource = new CancellationTokenSource();
WaitHelper.OnWaitEventHandler += Cancel;
thread = new Thread(() =>
{
try
{
this.Invoke(new Action(() =>
{
WaitHelper.BeginWait(this, false);
}));
while (true)
{
Debug.WriteLine(DateTime.Now);
Task.Delay(1000).Wait();
//此处可以自定义 然后传递到ProcessCommand
//WaitHelper.SendCmd(WaitEnum.Wait);
}
}
catch (ThreadAbortException ex)
{
}
});
thread.Start();
}
private void Cancel()
{
thread.Abort();
}
}
- 前面两个按钮主要是测试直接使用
SplashScreenManager
显示和关闭的。
Result
- 点击取消可以直接中断。
Review
- 这个看起来是不是很简单,而且代码也少,但是你知道我这个前辈之前实现用了多少代码吗?明明一个事件可以解决的中断问题,愣是写了四五个,跳来跳去的。不得不说,如果后人看到我写的代码会不会也有这种感觉?说不定以后一个字段就可以解决了。