我正在运行一个发送邮件的线程。我怎么知道一个线程已完成执行?
new Thread(x => SendMail(node.Attributes["id"].Value.ToString(), node["fname"].InnerText + " " + node["lname"].InnerText, 500, node["email"].InnerText)).Start();
答案 0 :(得分:3)
var thr = new Thread(x => SendMail(node.Attributes["id"].Value.ToString(), node["fname"].InnerText + " " + node["lname"].InnerText, 500, node["email"].InnerText));
thr.Start();
thr.Join();//In this place main thread will wait
答案 1 :(得分:2)
您应该保留对您创建的线程实例的引用,然后检查ThreadState
。您还可以检查IsAlive
属性以查看该线程当前是否正在执行。
答案 2 :(得分:2)
这取决于您的环境。例如,您可以通过使用任务并订阅延续来避免所有线程。
e.g。
var task = Task.Run(DoSomething)
.ContinueWith(a => Whatever())
或者使用C#5给出的便利关键字:
var task = await Task.Run(DoSomething);
Whatever();
如果你必须使用线程,我建议传递一个代表,当电子邮件完成后你只需要调用它:
// Outside the thread
private Action callback;
// Before starting the thread
callback = MyMethod/*Or a lambda if you want*/;
// In the thread action
Action<object> threadBody = x =>
{
SendMail(node.Attributes["id"].Value.ToString(), node["fname"].InnerText + " " + node["lname"].InnerText, 500, node["email"].InnerText);
callback();
};
答案 3 :(得分:2)
另一种变体是将ThreadPool.QueueUserWorkItem
与ManualResetEvent
结合使用:
如:
private void DoWork()
{
List<ManualResetEvent> events = new List<ManualResetEvent>();
//in case you need to loop through multiple email addresses
//use the foreach here, assuming that the items is a list.
//foreach(var item in items)
//{
var resetEvent = new ManualResetEvent(false);
ThreadPool.QueueUserWorkItem(arg =>
{
SendMail(node.Attributes["id"].Value.ToString(),
node["fname"].InnerText + " " + node["lname"].InnerText,
500, node["email"].InnerText);
resetEvent.Set();
});
events.Add(resetEvent);
//} <- closes the foreach loop
//WaitHandle.WaitAll waits for all the threads to finish.
WaitHandle.WaitAll(events.ToArray());
MessageBox.Show("Mails are sent", "Notification");
}
如果您希望遍历电子邮件地址列表或数组并为每个邮件单独启动一个帖子,这将特别有用。
在您的情况下,如果您想在等待发送邮件时做其他事情。您只需在后台线程中运行上面的代码,当消息显示时,您就知道工作已完成。
public void StartMailThread()
{
Thread myThread = new Thread(DoWork)
{
IsBackground = true,
Name = "MailThread"
};
myThread.Start();
}
虽然,使用线程启动线程池对我来说似乎有点奇怪。
答案 4 :(得分:1)
除了Andrew的答案,您可以使用BackgroundWorker它已经有RunWorkerCompleted事件 BackgroundWorker