Как получить возвращаемое значение при вызове BeginInvoke/Invoke в C#

у меня есть этот маленький метод, который должен быть потокобезопасным. Все работает пока я не хочу, чтобы возвращаемое значение вместо Void. Как получить возвращаемое значение при вызове BeginInvoke?

public static string readControlText(Control varControl) {
        if (varControl.InvokeRequired) {
            varControl.BeginInvoke(new MethodInvoker(() => readControlText(varControl)));
        } else {
            string varText = varControl.Text;
             return varText;
        }

    }

Edit: я думаю, что BeginInvoke не является nessecary в этом случае, поскольку мне нужно значение от GUI, прежде чем поток может продолжаться. Поэтому использование Invoke также хорошо. Просто не знаю, как использовать его в следующем примере для возврата значения.

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;
        }

    }

но не знаю как получите значение, используя этот код;)

6 ответов


вам нужно вызвать (), чтобы вы могли дождаться возвращения функции и получить ее возвращаемое значение. Вам также понадобится другой тип делегата. Это должно сработать:

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);
    }
}

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;
}

Если вы хотите получить возвращаемое значение от вашего метода, вы не должны использовать асинхронную версию метода, вы должны использовать .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);

delegate string StringInvoker();
    string GetControlText()
    {
        if (control.InvokeRequired)
        {
            string controltext = (string)control.Invoke(new StringInvoker(GetControlText));
            return(controltext);
        }
        else
        {
            return(control.Text);
        }
    }

/ / простой и элегантный, но необходимо дождаться другого потока для выполнения делегата; однако, если вы не можете продолжить без результатов...