1

我正在尝试通过 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] 调试 SerialConnections.M234Serial - 创建实例..

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

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

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

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

4

2 回答 2

1

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


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

AT^SCFG 命令显然是专有的制造商特定命令,因此我没有相应的文档参考。大多数由 3GPP 标准化的与手机相关的 AT 命令在27.007中给出,尽管一些(与 SMS 相关的)在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(您也可以isFinalResponseErrorST-Ericsson 的 U300 RILisFinalResponseSuccess进行比较,但请注意,这不是最终结果代码,而是中间结果代码,因此名称 isFinalResponseSuccess 是严格来说不是 100% 正确)。CONNECT


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

于 2015-07-28T22:55:30.267 回答
0

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

  • waitTime(System.currentTimeMillis(), 2000);在事件监听器中替换为if ( serialPortEvent.isRXCHAR() ) { ...
  • 确保正确终止 AT 命令;通常每个命令字符串的末尾都需要换行和/或回车。检查设备的文档。
  • 使isReceived volatile,即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();
  }
} 
于 2015-07-28T13:54:12.083 回答