Чтобы объяснить эту проблему, я поместил все необходимое в небольшое приложение, которое, как мы надеемся, объяснит проблему. Я действительно старался сделать все как можно меньше, но в моем реальном приложении эти разные актеры не знают друг друга и тоже не должны. Таким образом, простой ответ типа «возьмите переменную несколькими строками выше и вызовите для нее Invoke» не сработает.
Итак, давайте начнем с кода, а затем немного больше объяснений. Сначала есть простой класс, который реализует INotifyPropertyChanged:
public class MyData : INotifyPropertyChanged
{
private string _MyText;
public MyData()
{
_MyText = "Initial";
}
public string MyText
{
get { return _MyText; }
set
{
_MyText = value;
PropertyChanged(this, new PropertyChangedEventArgs("MyText"));
}
}
public event PropertyChangedEventHandler PropertyChanged;
}
Так что ничего особенного. А вот пример кода, который можно просто вставить в любой пустой проект консольного приложения:
static void Main(string[] args)
{
// Initialize the data and bindingSource
var myData = new MyData();
var bindingSource = new BindingSource();
bindingSource.DataSource = myData;
// Initialize the form and the controls of it ...
var form = new Form();
// ... the TextBox including data bind to it
var textBox = new TextBox();
textBox.DataBindings.Add("Text", bindingSource, "MyText");
textBox.DataBindings.DefaultDataSourceUpdateMode = DataSourceUpdateMode.OnPropertyChanged;
textBox.Dock = DockStyle.Top;
form.Controls.Add(textBox);
// ... the button and what happens on a click
var button = new Button();
button.Text = "Click me";
button.Dock = DockStyle.Top;
form.Controls.Add(button);
button.Click += (_, __) =>
{
// Create another thread that does something with the data object
var worker = new BackgroundWorker();
worker.RunWorkerCompleted += (___, ____) => button.Enabled = true;
worker.DoWork += (___, _____) =>
{
for (int i = 0; i < 10; i++)
{
// This leads to a cross-thread exception
// but all i'm doing is simply act on a property in
// my data and i can't see here that any gui is involved.
myData.MyText = "Try " + i;
}
};
button.Enabled = false;
worker.RunWorkerAsync();
};
form.ShowDialog();
}
Если вы запустите этот код, вы получите межпотоковое исключение, пытаясь изменить свойство MyText
. Это происходит, потому что объект MyData
вызывает PropertyChanged
, который будет перехвачен BindindSource
. Затем в соответствии с Binding
попытается обновить свойство Text
TextBox
. Что явно приводит к исключению.
Моя самая большая проблема здесь заключается в том, что объект MyData
не должен ничего знать о графическом интерфейсе (потому что это простой объект данных). Также рабочий поток ничего не знает о графическом интерфейсе. Он просто воздействует на кучу объектов данных и манипулирует ими.
ИМХО, я думаю, BindingSource
должен проверить, в каком потоке живет принимающий объект, и сделать соответствующее Invoke()
, чтобы получить значение их. К сожалению, это не встроено в него (или я ошибаюсь?), Поэтому мой вопрос:
Как разрешить это межпотоковое исключение, если объект данных или рабочий поток ничего не знают об источнике привязки, который прослушивает их события, чтобы отправить данные в графический интерфейс.