为什么我的异步方法不等待“serialPort = await SerialDevice.FromIdAsync()”?

时间:2017-01-15 18:27:13

标签: c# asynchronous raspberry-pi3 windows-10-iot-core

我正在尝试创建一个初始化树莓和arduino之间串行连接的类。该类还应该能够通过已建立的串行连接进行读写。

我使用的代码来自microsoft developers website

using Windows.Storage.Streams;
using Windows.Devices.Enumeration;
using Windows.Devices.SerialCommunication;

public async void Serial()
{
    /* Find the selector string for the serial device   */
    string aqs = SerialDevice.GetDeviceSelector("UART0");
    /* Find the serial device with our selector string  */
    var dis = await DeviceInformation.FindAllAsync(aqs);
    /* Create an serial device with our selected device */
    SerialDevice SerialPort = await SerialDevice.FromIdAsync(dis[0].Id);
    /* Configure serial settings */
    SerialPort.WriteTimeout = TimeSpan.FromMilliseconds(1000);
    SerialPort.ReadTimeout = TimeSpan.FromMilliseconds(1000);
    SerialPort.BaudRate = 9600;
    SerialPort.Parity = SerialParity.None;
    SerialPort.StopBits = SerialStopBitCount.One;
    SerialPort.DataBits = 8;

    /* Write a string out over serial */
    string txBuffer = "Hello Serial";
    DataWriter dataWriter = new DataWriter();
    dataWriter.WriteString(txBuffer);
    uint bytesWritten = await SerialPort.OutputStream.WriteAsync(dataWriter.DetachBuffer());

    /* Read data in from the serial port */
    const uint maxReadLength = 1024;
    DataReader dataReader = new DataReader(SerialPort.InputStream);
    uint bytesToRead = await dataReader.LoadAsync(maxReadLength);
    string rxBuffer = dataReader.ReadString(bytesToRead);
}

我将serialcommunication功能添加到Package.appxmanifest。

<Capabilities>
  <DeviceCapability Name="serialcommunication">
    <Device Id="any">
      <Function Type="name:serialPort" />
    </Device>
  </DeviceCapability>
</Capabilities>

到目前为止一切正常。 Raspberry在另一端成功地发送和接收来自Arduino的消息。

现在我尝试单独执行这些步骤。首先初始化串行连接,因此只要应用程序需要它们就可以使用读写函数。

的MainPage

namespace SerialUART
{
    public sealed partial class MainPage : Page
    {
        private SerialController serial = new SerialController();

        public MainPage()
        {
            this.InitializeComponent();
            serial.Write();
            serial.Read();
        }
    }
}

SerialController

using Windows.Devices.Enumeration;
using Windows.Devices.SerialCommunication;
using Windows.Storage.Streams;

namespace SerialUART
{
    class SerialController
    {
        private SerialDevice SerialPort;
        private DataWriter dataWriter;
        private DataReader dataReader;

        public SerialController()
        {
            InitSerial();
        }

        private async void InitSerial()
        {
            string aqs = SerialDevice.GetDeviceSelector("UART0");
            var dis = await DeviceInformation.FindAllAsync(aqs);
            SerialPort = await SerialDevice.FromIdAsync(dis[0].Id);    
            // The program does not wait here and executes Write()
            // after Write() the following code will be executed
            SerialPort.WriteTimeout = TimeSpan.FromMilliseconds(1000);
            SerialPort.ReadTimeout = TimeSpan.FromMilliseconds(1000);
            SerialPort.BaudRate = 9600;
            SerialPort.Parity = SerialParity.None;
            SerialPort.StopBits = SerialStopBitCount.One;
            SerialPort.DataBits = 8;
            dataWriter = new DataWriter();
            dataReader = new DataReader(SerialPort.InputStream);
        }

        public async void Read()
        {
            /* Read data in from the serial port */
            const uint maxReadLength = 1024;
            uint bytesToRead = await dataReader.LoadAsync(maxReadLength);
            string rxBuffer = dataReader.ReadString(bytesToRead);
            Debug.WriteLine(rxBuffer);
        }

        public async void Write()
        {
            /* Write a string out over serial */
            string txBuffer = "Hello Serial";
            dataWriter.WriteString(txBuffer);
            uint bytesWritten = await SerialPort.OutputStream.WriteAsync(dataWriter.DetachBuffer());
       }
    }
}

此代码不会等待serialPort,dataWriter和dataReader初始化。所以他们永远不会被分配到。

任何人都可以向我解释为什么它不等待串行连接?它应该如何完成?

2 个答案:

答案 0 :(得分:11)

所有async方法返回void - 这意味着它们开始执行,并且调用代码没有简单方法等待它实际完成之前持续。只要您对尚未完成的内容使用await表达式,异步方法将返回调用者,并安排了一个将在等待完成时执行的延续。在您的情况下,您只需继续下一个陈述。

相反,您应该让异步方法返回Task,并更改您的调用方法以等待异步方法完成后再继续。 优雅很难做到,因为在每种情况下,您都要从构造函数中调用异步方法,这些方法本身可能是异步的。您可能需要考虑使用异步静态方法而不是在构造函数中进行工作,构造函数可以调用异步方法,等待它们,然后调用一个不做任何实际工作的实际构造函数。

在您进行任何详细更改之前,您应该真正了解asyncawait所做的事情 - 尝试使用它们而不了解它们是进一步问题的一个方法。

答案 1 :(得分:1)

这是你的构造函数:

public SerialController()
{
    InitSerial();
}

它调用InitSerial()方法,其中包含多个等待方法。构造函数立即返回,而不执行所需的所有操作。然后你调用这些方法:

serial.Write();
serial.Read();

但是这些方法依赖于在InitSerial方法中执行的每一行代码,但显然它们没有。

解决问题的一种方法是让InitSerial不是私有的(内部或公共的)并让它返回Task。当你调用它时,你需要等待它,以便它已被执行完毕。之后,您可以致电serial.Write()Serial.Read()

你应该避免async void。它们仅用于事件处理。对于其他方法,请返回TaskTask<T>