当请求com口returns同样的请求

When requesting com port returns the same request

我正在尝试通过 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] DEBUG SerialConnections.M234Serial - 收到消息:AT^SCFG?

16:19:23.977 [主要] DEBUG SerialConnections.M234Serial - 结束

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

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

  • 将事件侦听器中的waitTime(System.currentTimeMillis(), 2000);替换为if ( serialPortEvent.isRXCHAR() ) { ...
  • 确保你正确地终止了你的AT命令;通常每个命令字符串的末尾都需要换行 and/or 回车 return。检查您的设备的文档。
  • 使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();
  }
} 

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


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

结束

AT^SCFG 命令显然是专有的制造商特定命令,因此我没有相关的文档参考。大多数由 3GPP 标准化的移动 phone 相关 AT 命令在 27.007, although some (SMS related) are given in 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 结尾的完整行,然后 returns 该行。

你可以看看atinout for an example for the is_final_result_code function (you can also compare to isFinalResponseError and isFinalResponseSuccess in ST-Ericsson's U300 RIL的代码,虽然注意CONNECT不是最终结果代码,它是一个中间结果代码,所以名称是FinalResponseSuccess严格来说不是100%正确)。


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