Java/Arduino - чтение данных с последовательного порта
У меня есть программа на Java, где я должен прочитать информацию, которую посылает Arduino. Я взял код Java отсюда. Теперь я не совсем понял, как это работает, но я попытался изменить его, и я получил это:
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.io.OutputStream;
import gnu.io.CommPortIdentifier;
import gnu.io.SerialPort;
import gnu.io.SerialPortEvent;
import gnu.io.SerialPortEventListener;
import java.util.Enumeration;
public class Serial implements SerialPortEventListener {
SerialPort serialPort;
private static final String PORT_NAMES[] = {
"/dev/tty.usbserial-A9007UX1", // Mac OS X
"/dev/ttyUSB0", // Linux
"COM3", // Windows
};
private BufferedReader input;
private static OutputStream output;
private static final int TIME_OUT = 2000;
private static final int DATA_RATE = 115200;
public void initialize() {
CommPortIdentifier portId = null;
Enumeration portEnum = CommPortIdentifier.getPortIdentifiers();
while (portEnum.hasMoreElements()) {
CommPortIdentifier currPortId = (CommPortIdentifier) portEnum.nextElement();
for (String portName : PORT_NAMES) {
if (currPortId.getName().equals(portName)) {
portId = currPortId;
break;
}
}
}
if (portId == null) {
System.out.println("Could not find COM port.");
return;
}
try {
serialPort = (SerialPort) portId.open(this.getClass().getName(),TIME_OUT);
serialPort.setSerialPortParams(DATA_RATE, SerialPort.DATABITS_8, SerialPort.STOPBITS_1, SerialPort.PARITY_NONE);
input = new BufferedReader(new InputStreamReader(serialPort.getInputStream()));
output = serialPort.getOutputStream();
serialPort.addEventListener(this);
serialPort.notifyOnDataAvailable(true);
}
catch (Exception e) {
System.err.println(e.toString());
}
}
public synchronized void serialEvent(SerialPortEvent oEvent) {
if (oEvent.getEventType() == SerialPortEvent.DATA_AVAILABLE) {
try {
String inputLine=input.readLine();
System.out.println(inputLine);
} catch (Exception e) {
System.err.println(e.toString());
}
}
}
public synchronized void close() {
if (serialPort != null) {
serialPort.removeEventListener();
serialPort.close();
}
}
public Serial(String ncom){
if(Integer.parseInt(ncom)>=3 && Integer.parseInt(ncom)<=9)
PORT_NAMES[2] = "COM" + ncom;
initialize();
Thread t=new Thread() {
public void run() {
try {Thread.sleep(1000000);} catch (InterruptedException ie) {}
}
};
t.start();
System.out.println("Serial Comms Started");
}
public synchronized void send(int b){
try{
output.write(b);
}
catch (Exception e) {
System.err.println(e.toString());
}
}
public synchronized int read(){
int b = 0;
try{
b = (int)input.read();
}
catch (Exception e) {
System.err.println(e.toString());
}
return b;
}
}
Я создаю объект Serial с нужным мне COM-портом в основной программе, затем использую Serial.read
а также Serial.write
когда мне это нужно.
Serial.write
отлично работает, Arduino получает данные и показывает их на ЖК-дисплее. Проблема в Serial.read
, Когда программа запущена, она продолжает чтение с последовательного порта (примерно каждые 40 мс), но это не значит, что Arduino что-то послал. Arduino отправляет байт только при нажатии кнопки. Таким образом, когда код Java работает, он выдает "n" исключение перед чтением чего-либо, и это вызывает такое большое отставание.
Я знаю, мне нужно что-то вроде Serial.available()
, Я старался input.available()
, но это не работает. Я не знаю, как решить эту проблему.
Если у вас есть код, который работает, я был бы очень признателен, если бы вы могли дать его мне. Мне просто нужно два метода, читать и писать, мне все равно, как работает код:D
РЕДАКТИРОВАТЬ:
Я изменил класс Serial, теперь он снова имеет этот метод, как сказал apremalal
public synchronized void serialEvent(SerialPortEvent oEvent) {
if (oEvent.getEventType() == SerialPortEvent.DATA_AVAILABLE) {
try {
String inputLine=null;
if (input.ready()) {
inputLine = input.readLine();
panel.read(inputLine);
}
} catch (Exception e) {
System.err.println(e.toString());
}
}
}
а в другом классе (в данном случае Panel) я получил это:
public void read(String data){
System.out.println(data);
System.out.println(data == "255");
if(data == "255")
//code here
}
Он печатает значения правильно, но data == "255"
всегда ложно, даже если я действительно получаю 255
.... Я пытался сделать Integer.parseInt
но ничего не изменилось. Почему, черт возьми?
РЕДАКТИРОВАТЬ 2: Хорошо решено:\
public void read(String data){
serialRead = Integer.parseInt(data);
if(serialRead == 255)
//code here
}
Теперь это работа.. не знаю, почему я должен был сделать это... что угодно:)
2 ответа
Вы не хотите специально писать функцию чтения, она уже есть в примере кода. Как отметил TheMerovingian, вы можете проверить входной буфер перед чтением. Вот рабочий код, который я использовал в одном из моих проектов.
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.io.OutputStream;
import gnu.io.CommPortIdentifier;
import gnu.io.SerialPort;
import gnu.io.SerialPortEvent;
import gnu.io.SerialPortEventListener;
import java.util.Enumeration;
public class SerialTest implements SerialPortEventListener {
SerialPort serialPort;
/** The port we're normally going to use. */
private static final String PORT_NAMES[] = { "/dev/tty.usbserial-A9007UX1", // Mac OS X
"/dev/ttyUSB0", // Linux
"COM35", // Windows
};
private BufferedReader input;
private OutputStream output;
private static final int TIME_OUT = 2000;
private static final int DATA_RATE = 9600;
public void initialize() {
CommPortIdentifier portId = null;
Enumeration portEnum = CommPortIdentifier.getPortIdentifiers();
//First, Find an instance of serial port as set in PORT_NAMES.
while (portEnum.hasMoreElements()) {
CommPortIdentifier currPortId = (CommPortIdentifier) portEnum.nextElement();
for (String portName : PORT_NAMES) {
if (currPortId.getName().equals(portName)) {
portId = currPortId;
break;
}
}
}
if (portId == null) {
System.out.println("Could not find COM port.");
return;
}
try {
serialPort = (SerialPort) portId.open(this.getClass().getName(),
TIME_OUT);
serialPort.setSerialPortParams(DATA_RATE,
SerialPort.DATABITS_8,
SerialPort.STOPBITS_1,
SerialPort.PARITY_NONE);
// open the streams
input = new BufferedReader(new InputStreamReader(serialPort.getInputStream()));
output = serialPort.getOutputStream();
serialPort.addEventListener(this);
serialPort.notifyOnDataAvailable(true);
} catch (Exception e) {
System.err.println(e.toString());
}
}
public synchronized void close() {
if (serialPort != null) {
serialPort.removeEventListener();
serialPort.close();
}
}
public synchronized void serialEvent(SerialPortEvent oEvent) {
if (oEvent.getEventType() == SerialPortEvent.DATA_AVAILABLE) {
try {
String inputLine=null;
if (input.ready()) {
inputLine = input.readLine();
System.out.println(inputLine);
}
} catch (Exception e) {
System.err.println(e.toString());
}
}
// Ignore all the other eventTypes, but you should consider the other ones.
}
public static void main(String[] args) throws Exception {
SerialTest main = new SerialTest();
main.initialize();
Thread t=new Thread() {
public void run() {
//the following line will keep this app alive for 1000 seconds,
//waiting for events to occur and responding to them (printing incoming messages to console).
try {Thread.sleep(1000000);} catch (InterruptedException ie) {}
}
};
t.start();
System.out.println("Started");
}
}
РЕДАКТИРОВАТЬ: функция serialEvent отвечает за чтение буфера.
public synchronized void serialEvent(SerialPortEvent oEvent) {
if (oEvent.getEventType() == SerialPortEvent.DATA_AVAILABLE) {
try {
String inputLine=null;
if (input.ready()) {
inputLine = input.readLine();
System.out.println(inputLine);
}
} catch (Exception e) {
System.err.println(e.toString());
}
}
// Ignore all the other eventTypes, but you should consider the other ones.
}
BufferedReader
класс имеет ready()
метод, который возвращает True
if "буфер не пустой, или если основной символьный поток готов". а также False
иначе. Таким образом, вы можете добавить чек в read()
метод, чтобы убедиться, что есть данные для чтения, прежде чем пытаться читать.
public synchronized int read(){
int b = 0;
try{
if (input.ready()) {
b = (int)input.read();
}
}catch (Exception e) {
System.err.println(e.toString());
}
return b;
}
Похоже, код имеет try-catch
на месте, чтобы справиться, если эти вещи терпят неудачу, что, возможно, является причиной вашего отставания, потому что try-catch
довольно дорогие. Итак input.ready()
проверка должна привести к меньшему количеству исключений.