调用线程无法访问此对象,因为其他线程拥有它.例外

本文关键字:线程 其他 因为 拥有 例外 对象 访问 调用 | 更新日期: 2023-09-27 18:35:06

private void Thread1_Exe()
{
    try
    {
        int sleepValT1 = Convert.ToInt32(listBoxT2.SelectedValue);
        int StartLoop = 0;
        int EndLoop = 10000;
        for (int i = StartLoop; i <= EndLoop; i++)
        {
            Dispatcher.BeginInvoke(
            new Action(() => listboxE1.Items.Add("T1: Execution Count> " + i.ToString())));
            Thread.Sleep(sleepValT1);
        }
    }
    catch (Exception Ex)
    {
        MessageBox.Show(Ex.Message);
    }
}

我试图在不同的线程上调用上述函数

private void thread1_Click(object sender, RoutedEventArgs e)
{
    threadBtn1.IsEnabled = false;
    Thread t1 = new Thread(new ThreadStart(Thread1_Exe));
    t1.Start();
}

但是当我从列表框中选择值并尝试将其保存在变量中,然后尝试将该值传入时,会发生异常

Thread.Sleep()

我总是得到这个异常,不同的线程拥有这个对象。尝试了很多事情。请帮助我找出我做错的地方。

谢谢

调用线程无法访问此对象,因为其他线程拥有它.例外

对 UI

元素的任何访问,无论是读取还是写入,都必须通过主 UI 线程调用。此线程可通过 Dispatcher 访问。

在您的代码中,我看到一行情况并非如此:当您调用listBoxT2.SelectedValue

要修复错误,请在Dispatcher线程上调用此代码并将其保存到局部变量:

int selectedValue;
listBoxT2.Dispatcher.Invoke(() => selectedValue = Convert.ToInt32(listBoxT2.SelectedValue));
   

代码的其余部分应该可以正常工作。

您的问题是您正在从后台线程访问 UI 控件。UI 往往是单线程的,因此需要协调对控件的访问,以便始终通过 UI 线程进行访问。

您可以做的最简单的事情是在启动新线程之前从列表框中读取值:

private void Thread1_Exe(int sleepVal)
{
    try
    {
        int StartLoop = 0;
        int EndLoop = 10000;

        for (int i = StartLoop; i <= EndLoop; i++)
        {
            Dispatcher.BeginInvoke(
            new Action(() =>
            listboxE1.Items.Add("T1: Execution Count> " + i.ToString())));
            Thread.Sleep(sleepVal);
        }
    }
    catch (Exception Ex)
    {
        MessageBox.Show(Ex.Message);
    }
}
private void thread1_Click(object sender, RoutedEventArgs e)
{
    threadBtn1.IsEnabled = false;
    int sleepValT1 = Convert.ToInt32(listBoxT1.SelectedValue);
    Thread t1 = new Thread(() => Thread1_Exe(sleepValT1));
    t1.Start();
}

其他替代方法包括通过其调度程序线程访问控件:

private void Thread1_Exe()
{
    try
    {
        int sleepValT1;
        listBoxT1.Dispatcher.Invoke(() => sleepValT1 = Convert.ToInt32(listBoxT1.SelectedValue));
        int StartLoop = 0;
        int EndLoop = 10000;

        for (int i = StartLoop; i <= EndLoop; i++)
        {
            Dispatcher.BeginInvoke(
            new Action(() =>
            listboxE1.Items.Add("T1: Execution Count> " + i.ToString())));
            Thread.Sleep(sleepValT1);
        }
    }
    catch (Exception Ex)
    {
        MessageBox.Show(Ex.Message);
    }
}
private void thread1_Click(object sender, RoutedEventArgs e)
{
    threadBtn1.IsEnabled = false;
    Thread t1 = new Thread(() => Thread1_Exe());
    t1.Start();
}

通常,您确实希望尽可能避免在线程之间共享可变状态,以使事情更容易推理。