当请求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。检查您的设备的文档。
- 使
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();
}
}
像 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
中给出
如开头所述,需要更改结构。您应该 永远,永远,永远,永远 使用 waitTime
、sleep
或任何类似的东西来等待调制解调器的响应。这就像踢挡路的狗以让它们移动一样有用。是的,你可能很幸运,有时它确实有效,但在某些时候你会后悔采用这种方法......
唯一可靠的方法是做类似于
的事情
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
命令禁用,但是使用像上面这样的适当解析结构,这通常无关紧要,因为您只是将回显命令作为将被忽略的行读取。
我正在尝试通过 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。检查您的设备的文档。
- 使
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();
}
}
像 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
中给出如开头所述,需要更改结构。您应该 永远,永远,永远,永远 使用 waitTime
、sleep
或任何类似的东西来等待调制解调器的响应。这就像踢挡路的狗以让它们移动一样有用。是的,你可能很幸运,有时它确实有效,但在某些时候你会后悔采用这种方法......
唯一可靠的方法是做类似于
的事情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
命令禁用,但是使用像上面这样的适当解析结构,这通常无关紧要,因为您只是将回显命令作为将被忽略的行读取。