У меня есть этот небольшой метод, который, как предполагается, ориентирован на многопотоковое исполнение. Все работает, пока я не хочу, чтобы это имело возвращаемое значение вместо пустоты. Как я получаю возвращаемое значение, когда BeginInvoke называют?
public static string readControlText(Control varControl) {
if (varControl.InvokeRequired) {
varControl.BeginInvoke(new MethodInvoker(() => readControlText(varControl)));
} else {
string varText = varControl.Text;
return varText;
}
}
Править: Я предполагаю наличие, BeginInvoke не является nessecary в этом случае, поскольку мне нужно значение от GUI, прежде чем поток сможет продолжиться. Так использование Вызывает, хорошо также. Просто никакая подсказка, как использовать его в следовании примеру к возвращаемому значению.
private delegate string ControlTextRead(Control varControl);
public static string readControlText(Control varControl) {
if (varControl.InvokeRequired) {
varControl.Invoke(new ControlTextRead(readControlText), new object[] {varControl});
} else {
string varText = varControl.Text;
return varText;
}
}
Но не уверенный, как получить значение с помощью того кода также ;)
Вам нужно Invoke(), чтобы вы могли дождаться возврата функции и получить ее возвращаемое значение. Вам также понадобится другой тип делегата. Это должно сработать:
public static string readControlText(Control varControl) {
if (varControl.InvokeRequired) {
return (string)varControl.Invoke(
new Func<String>(() => readControlText(varControl))
);
}
else {
string varText = varControl.Text;
return varText;
}
}
EndInvoke
может использоваться для получения возвращаемого значения из вызова BeginInvoke
. Например:
public static void Main()
{
// The asynchronous method puts the thread id here.
int threadId;
// Create an instance of the test class.
AsyncDemo ad = new AsyncDemo();
// Create the delegate.
AsyncMethodCaller caller = new AsyncMethodCaller(ad.TestMethod);
// Initiate the asychronous call.
IAsyncResult result = caller.BeginInvoke(3000,
out threadId, null, null);
Thread.Sleep(0);
Console.WriteLine("Main thread {0} does some work.",
Thread.CurrentThread.ManagedThreadId);
// Call EndInvoke to wait for the asynchronous call to complete,
// and to retrieve the results.
string returnValue = caller.EndInvoke(out threadId, result);
Console.WriteLine("The call executed on thread {0}, with return value \"{1}\".",
threadId, returnValue);
}
}
Если вы хотите получить возвращаемое значение от вашего метода, вы не должны использовать асинхронную версию метода, вы должны использовать .Invoke (...)
. Это синхронно, то есть он выполнит ваш делегат и не вернется, пока он не будет завершен. В вашем примере, как сейчас, BeginInvoke отправит запрос на выполнение вашего делегата и сразу же вернется. Так что возвращать нечего.
Вы хотели что-то подобное?
// begin execution asynchronously
IAsyncResult result = myObject.BeginInvoke("data.dat", null, null);
// wait for it to complete
while (result.IsCompleted == false) {
// do some work
Thread.Sleep(10);
}
// get the return value
int returnValue = myObject.EndInvoke(result);
public static string readControlText(Control varControl)
{
if (varControl.InvokeRequired)
{
string res = "";
var action = new Action<Control>(c => res = c.Text);
varControl.Invoke(action, varControl);
return res;
}
string varText = varControl.Text;
return varText;
}