1

我正在开发一个轮询 RS232 到 I2C 转换器的应用程序,我连接了一个 i2c 设备,它必须实时轮询,这意味着不断:D 一旦响应到来,我再次重新发出命令......

我的代码:

        lock (_locker)
        {
            try
            {

                _serialPort.DiscardInBuffer();
                _serialPort.Write(sendArray, 0, sendArray.Length);
                _serialPort.ReadFixed(headerArray, headerArray.Length);
                returnOperation = (DeviceOperation)((int)headerArray[1] << 8 | (int)headerArray[3]);
                DataToReceive = headerArray[5] * 256 + headerArray[6];
                if (DataToReceive != 0)
                {
                    _serialPort.ReadFixed(recvArray, DataToReceive);
                }
                _serialPort.ReadFixed(EOT, 1); //EOT
            }
            catch (Exception e )
            {
                Logger.Log(Level.Error, _name, "Fauled to execute command.", e);
                return false;                    
            }
        }

其中 ReadFixed 是一个扩展:

    public static void ReadFixed(this SerialPort port, byte[] buffer, int count)
    {
        int offset = 0;
        while (count != offset)
            offset += port.Read(buffer, offset, count - offset);

        if (count != offset)
            throw new TimeoutException("ReadFixed did not receive specified bytes in time!");
    }

这段代码在双核处理器上导致大约 40 - 60 % 的 CPU 使用率。我用了ANTS profiler,它说函数READFIXED很热,里面消耗的cpu真的很高。

知道有什么问题吗?为什么这样的cpu使用率?难道我做错了什么 ?

谢谢您的回复!

4

1 回答 1

3

默认情况下serialport.Read()将立即返回,无论是否接收到字节。如果设置serialport.Timeout = 2000,线程将阻塞,直到字节可用或在 2 秒后超时。这将导致 CPU 使用率降低,并且将提供比Thread.Sleep.

请注意,这将阻止您的线程。如果这是一个问题,请在一个BackgroundWorker线程中完成所有阅读。您可以使用信号,例如使用AutoResetEvent,来有效地与前台线程通信。

于 2014-11-02T14:05:11.110 回答