不确定如何在线程之间传递对象

本文关键字:对象 之间 线程 不确定 | 更新日期: 2023-09-27 18:36:47

我意识到我之前问了一个非常相似的问题,但我的结构是错误的。我错误地认为我可以在调用中执行我的图标生成。这导致了一个不同的问题。

我有一个包含 500 SVG 的文件夹。我想在文件夹中创建每个 SVG 的对象。我需要在单独的线程上执行此操作,因为它可能需要一些时间才能完成,并且正在锁定 UI。

private void Grid_Loaded(object sender, RoutedEventArgs e)
{
     Thread t = new Thread(LoadIcons);
     t.SetApartmentState(ApartmentState.STA);
     t.Start();
}
private void LoadIcons()
{
   //Populate ListOfSVGsInFolder
   Foreach(String SVGFile in ListOfSVGsInFolder)
   {
       Icon icon = new Icon
       //Perform ~50 lines of code which get the paths and other details from the
       //SVGFile and plug them into my icon object
       //Now I had a fully generated Icon
       //Add the icon to the form
       WrapPanel.Children.Add(icon)
   }
} 

我的问题是我无法将图标添加到包装面板。由于我希望此代码在单独的线程上执行,因此我无法直接与UI对话。但是,我可以做到:

Foreach(String SVGFile in ListOfSVGsInFolder)
{         
    Icon icon = new Icon
    //Perform ~50 lines of code which get the paths and other details from the
    //SVGFile and plug them into my icon object
    Dispatcher.Invoke(new Action(() =>
    {         
      WrapPanel.Children.Add(icon);
    }));
}

但这样做,我现在在尝试将我的图标对象添加到 WrapPanel 时无法再访问它。

基本上,我希望能够在文件夹中找到的 SVG 上执行所有这些计算,在同一线程中创建 SVG 的对象,然后将这些对象添加到 UI 中。

不确定如何在线程之间传递对象

Thread t = new Thread(LoadIcons); //Don't do this

通常不要创建线程来执行后台工作。对于系统和您自己来说,正确管理它们都需要做很多工作。请改用 ThreadPool。

最简单的方法是使用TaskFactory

foreach(string svgFile in listOfSVGsInFolder)
{
    Task.Run(() => // Task.Factory.StartNew for pre .net 4.5
      {
        Debug.WriteLine ("Creating SVG in thread {0}", Thread.CurrentThread.ManagedThreadId);
        Icon icon = // whatever you do to create it
        Application.Current.Dispatcher.BeginInvoke(
          DispatcherPriority.Background,
          () => {         
                  WrapPanel.Children.Add(icon);
                });
      });
}

这就是为什么有BackgroundWorker

private void Grid_Loaded(object sender, RoutedEventArgs e)
{
    BackgroundWorker bgWorker = new BackgroundWorker();
    bgWorker.DoWork += LoadIcons;
    bgWorker.ProgressChanged += IconDone;
    bgWorker.RunWorkerAsync();
}
private void IconDone(object sender, ProgressChangedEventArgs e)
{
    Icon icon = e.UserState as Icon;
    if (icon != null)
        WrapPanel.Children.Add(icon); //This code is executed in the GUI thread
}
private void LoadIcons(object sender, DoWorkEventArgs doWorkEventArgs)
{
    BackgroundWorker worker = sender as BackgroundWorker;
    //Populate ListOfSVGsInFolder
    foreach (String SVGFile in ListOfSVGsInFolder)
    {
        Icon icon = new Icon
        //Perform ~50 lines of code which get the paths and other details from the
        //SVGFile and plug them into my icon object
        //Now I had a fully generated Icon

        worker.ReportProgress(0, icon);
    }
}

更多信息 : MSDN

你需要一个图标列表:

List<Icon> iconList = new List<Icon>();

在加载图标中:

// ... build your icon here
lock(iconList)
{
    iconList.Add(icon);
}

在 UI 线程中:

lock(iconList)
{
    icon = iconList[0];
}
// use the icon in the GUI

当然,您必须在UI线程中检查列表中是否有内容,并从此列表中删除"已使用"图标,所有这些都在lock()中