【问题标题】:When requesting com port returns the same request请求com端口时返回相同的请求
【发布时间】:2015-07-28 13:30:18
【问题描述】:

我正在尝试通过 COM 端口发送 AT 命令,但只接收到相同的命令。

package SerialConnections;

import jssc.SerialPort;
import jssc.SerialPortEvent;
import jssc.SerialPortEventListener;
import jssc.SerialPortException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import static ru.telemetria.qa.utils.Utilities.waitTime;

public class M234Serial {
    private static Logger log = LoggerFactory.getLogger(M234Serial.class);
    private SerialPort serialPort;
    private byte[] receivedData;
    private boolean isReceived;

    public M234Serial() throws Exception {

        serialPort = new SerialPort("COM55");
    }

    public void sendCommand() throws Exception {
        open();

        String command = "AT^SCFG?";
        serialPort.writeBytes(command.getBytes());
        log.debug("Send request: " + command);

        while (!isReceived) {}

        close();
    }

    private void open() throws Exception {
        serialPort.openPort();
        serialPort.setParams(SerialPort.BAUDRATE_115200, SerialPort.DATABITS_8, SerialPort.STOPBITS_1, SerialPort.PARITY_NONE);
        serialPort.addEventListener(new SerialPortEventListener() {
            @Override
            public void serialEvent(SerialPortEvent serialPortEvent) {
                try {
                    waitTime(System.currentTimeMillis(), 2000);
                    receivedData = serialPort.readBytes();
                    log.debug("Received message: " + StringUtils.asciiToString(receivedData));
                    isReceived = true;
                    serialPort.removeEventListener();
                } catch (SerialPortException spe) {
                    spe.printStackTrace();
                }
            }
        });
    }

    private void close() throws Exception {
        serialPort.closePort();
    }

    public static void main(String[] args) throws Exception {
        log.debug("Create instance..");
        M234Serial serial = new M234Serial();
        serial.sendCommand();
        log.debug("End");
    }
}

日志:

16:19:21.910 [main] DEBUG SerialConnections.M234Serial - 创建实例..

16:19:21.974 [main] DEBUG SerialConnections.M234Serial - 发送请求:AT^SCFG?

16:19:23.976 [EventThread COM55] 调试 SerialConnections.M234Serial - 收到消息:AT^SCFG?

16:19:23.977 [main] 调试 SerialConnections.M234Serial - 结束

我做错了什么,我该如何解决?

【问题讨论】:

  • 是否尝试在输入上读取更长的时间,然后才收到第一个事件?
  • 是的,我等了很久。但什么都没有发生

标签: java serial-port at-command jssc


【解决方案1】:

while (!isReceived) {} 这样的忙碌等待会产生可怕的性能,因此如果保持这种结构,您应该将变量从布尔值更改为互斥体/信号量或类似的东西。但你不应该保留它,所以我提这个仅供参考。


从获取V.250 调制解调器标准的副本开始,并至少阅读第 5 章的全部内容。这将教您很多基本的 AT 命令处理,例如应该以 @987654327 终止 AT 命令行@。

AT^SCFG 命令显然是专有的制造商特定命令,因此我没有相关的文档参考。大多数由 3GPP 标准化的手机相关 AT 命令在27.007 中给出,尽管一些(与短信相关的)在27.005 中给出

如开头所述,需要更改结构。您应该从不、从不、从不、从不使用waitTimesleep 或任何类似的东西来等待调制解调器的响应。它就像踢狗一样有用,以使它们移动。是的,您可能很幸运,并且有时它确实有效,但是在某些时候您会为采用这种方法而感到抱歉...

唯一可靠的方法是做类似的事情

serialPort.openPort();
...
// start sending AT^SCFG?
serialPort.writeBytes("AT^SCFG?\r");
do {
    line = readLine(serialPort);
} while (! is_final_result_code(line))
// Sending of AT^SCFG? command finished (successfully or not)
...
serialPort.closePort();

readLine 函数从串行端口读取一个字节和一个字节,直到它收到以\r\n 结尾的完整行,然后返回该行。

您可以查看atinout 的代码以获取is_final_result_code 函数的示例(您也可以比较ST-Ericsson's U300 RIL 中的isFinalResponseErrorisFinalResponseSuccess,尽管请注意CONNECT 不是最终的结果码,它是一个中间结果码,所以名字 isFinalResponseSuccess 严格来说不是 100% 正确)。


命令发送被接收回的问题与调制解调器回显命令有关。这可以通过ATE 命令禁用,但使用像上面这样的正确解析结构,这通常无关紧要,因为您只需将回显命令读取为将被忽略的行。

【讨论】:

  • 比你!是的,我知道睡眠等待线程方法。我会努力改进我的代码。
  • 您对“\r”终止的AT命令的评论非常有用 - 它有帮助!
【解决方案2】:

我建议对您的代码进行以下改进:

  • 将事件侦听器中的waitTime(System.currentTimeMillis(), 2000); 替换为if ( serialPortEvent.isRXCHAR() ) { ...
  • 确保正确终止 AT 命令;通常每个命令字符串的末尾都需要换行和/或回车。检查您设备的文档。
  • 使isReceivedvolatile,即private volatile boolean isReceived;,如果要在不同线程之间共享。

为避免忙于等待,您可以使用标准 Java 同步原语,如下所示:

private volatile boolean isReceived;

private final Object syncObject = new Object();

// ...

private void waitForReceived() {
  synchronized(syncObject) {
    while( !isReceived ) {
      syncObject.wait();
    }
  }
}


private void signalReceived() {
  synchronized(syncObject) {
    isReceived = true;
    syncObject.notifyAll();
  }
} 

【讨论】:

  • AT 命令行只能以\r 终止(前提是您没有弄乱您不应该这样做的S3 寄存器),并且绝不能使用任何其他EOL 变体。请参阅V.250 中的“5.2.1 命令行通用格式”一章。
  • 感谢volatile
猜你喜欢
  • 2019-09-10
  • 1970-01-01
  • 1970-01-01
  • 2019-05-07
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2012-07-16
相关资源
最近更新 更多