我有一些代码,我需要运行在不同的线程,而不是GUI,因为它目前导致窗体冻结,而代码运行(10秒左右)。

假设我以前从未创建过一个新线程;有什么简单/基本的例子,如何做到这一点在c#和使用。net Framework 2.0或更高版本?


当前回答

如果你想得到一个值:

var someValue;

Thread thread = new Thread(delegate()
            {                 
                //Do somthing and set your value
                someValue = "Hello World";
            });

thread.Start();

while (thread.IsAlive)
  Application.DoEvents();

其他回答

在. net中有许多运行独立线程的方法,每种方法都有不同的行为。你需要在GUI退出后继续运行线程吗?您需要在线程和GUI之间传递信息吗?线程是否需要更新GUI?线程应该完成一个任务然后退出,还是继续运行?这些问题的答案会告诉你该使用哪种方法。

在Code Project网站上有一篇很好的异步方法文章,描述了各种方法并提供了示例代码。

注意,本文是在async/await模式和任务并行库引入。net之前编写的。

如何:使用后台线程搜索文件

您必须非常小心从其他线程访问特定于GUI的东西(这对许多GUI工具包来说很常见)。如果你想在GUI中从处理线程中更新一些东西,检查这个答案,我认为这对WinForms很有用。对于WPF,看到这个(它展示了如何在UpdateProgress()方法中触摸组件,因此它将从其他线程工作,但实际上我不喜欢它在通过Dispathcer做BeginInvoke之前不做CheckAccess(),看到并搜索CheckAccess)

正在寻找。net专门关于线程的书,发现了这本(免费下载)。更多细节请参见http://www.albahari.com/threading/。

我相信你会发现你需要启动执行作为新线程在前20页,它有更多(不确定GUI特定片段,我的意思是严格特定于线程)。很高兴听到社区对这项工作的看法,因为我正在读这个。对我来说,现在看起来非常整洁(用于显示。net特定的线程方法和类型)。此外,它还涵盖了我真正欣赏的。net 2.0(而不是古老的1.1)。

另一种选择,使用委托和线程池…

假设'GetEnergyUsage'是一个方法,它接受一个DateTime和另一个DateTime作为输入参数,并返回一个Int…

// following declaration of delegate ,,,
public delegate long GetEnergyUsageDelegate(DateTime lastRunTime, 
                                            DateTime procDateTime);

// following inside of some client method 
GetEnergyUsageDelegate nrgDel = GetEnergyUsage;                     
IAsyncResult aR = nrgDel.BeginInvoke(lastRunTime, procDT, null, null);
while (!aR.IsCompleted) Thread.Sleep(500);
int usageCnt = nrgDel.EndInvoke(aR);

如果你打算使用原始线程对象,那么你需要将IsBackground设置为true,你还应该设置Threading Apartment模型(可能是STA)。

public static void DoWork()
{
    // do some work
}

public static void StartWorker()
{
    Thread worker = new Thread(DoWork);
    worker.IsBackground = true;
    worker.SetApartmentState(System.Threading.ApartmentState.STA);
    worker.Start()
}

如果你需要UI交互,我会推荐BackgroundWorker类。

将该代码放入一个函数中(不能与GUI在同一个线程上执行的代码),并执行以下代码来触发该代码的执行。

线程myThread=新线程(namefunction);

workerThread.Start();

在线程对象上调用start函数将导致在新线程中执行函数调用。