调用线程无法访问此对象,因为其他线程拥有它.例外
本文关键字:线程 其他 因为 拥有 例外 对象 访问 调用 | 更新日期: 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();
}
通常,您确实希望尽可能避免在线程之间共享可变状态,以使事情更容易推理。