Android串口使用2之使用Google官方库android-serialport-api

2023-12-15 10:59

本文主要是介绍Android串口使用2之使用Google官方库android-serialport-api,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Google官方库android-serialport-api

GitHub:android-serialport-api
此库版本太旧了,而且还不是AS工程,也不支持奇偶检验、数据位和停止位设定。

想要这个库支持奇偶检验、数据位和停止位设定也很简单

  1. 修改SerialPort.h和SerialPort.c两个文件
/SerialPort.h/
/** Class:     android_serialport_api_SerialPort* Method:    open* Signature: (Ljava/lang/String;II)Ljava/io/FileDescriptor;*/
JNIEXPORT jobject JNICALL Java_android_1serialport_1api_SerialPort_open(JNIEnv *, jclass, jstring, jint, jint, jint, jint);
/SerialPort.c/
/** Class:     android_serialport_SerialPort* Method:    open* Signature: (Ljava/lang/String;II)Ljava/io/FileDescriptor;*/
JNIEXPORT jobject JNICALL Java_android_1serialport_1api_SerialPort_open(JNIEnv *env, jclass thiz, jstring path,  jint baudrate, jint parity, jint dataBits, jint stopBits){int fd;int flags;speed_t speed;jobject mFileDescriptor;flags = 0;/* Check arguments */{speed = getBaudrate(baudrate);if (speed == -1) {/* TODO: throw an exception */LOGE("Invalid baudrate");return NULL;}if (parity < 0 || parity > 2) {LOGE("Invalid parity");return NULL;}if (dataBits < 5 || dataBits > 8) {LOGE("Invalid dataBits");return NULL;}if (stopBits < 1 || stopBits > 2) {LOGE("Invalid stopBit");return NULL;}}/* Opening device */{jboolean iscopy;const char *path_utf = (*env)->GetStringUTFChars(env, path, &iscopy);LOGD("Opening serial port %s with flags 0x%x", path_utf, O_RDWR | flags);fd = open(path_utf, O_RDWR | flags);LOGD("open() fd = %d", fd);(*env)->ReleaseStringUTFChars(env, path, path_utf);if (fd == -1){/* Throw an exception */LOGE("Cannot open port");/* TODO: throw an exception */return NULL;}}/* Configure device */{struct termios cfg;LOGD("Configuring serial port");if (tcgetattr(fd, &cfg)){LOGE("tcgetattr() failed");close(fd);/* TODO: throw an exception */return NULL;}cfmakeraw(&cfg);cfsetispeed(&cfg, speed);cfsetospeed(&cfg, speed);/* More attribute set */switch (parity) {case 0:cfg.c_cflag &= ~PARENB;    //无奇偶校验break;case 1:cfg.c_cflag |= (PARODD | PARENB);   //奇校验break;case 2:cfg.c_iflag &= ~(IGNPAR | PARMRK); // 偶校验cfg.c_iflag |= INPCK;cfg.c_cflag |= PARENB;cfg.c_cflag &= ~PARODD;break;default:cfg.c_cflag &= ~PARENB;break;}switch (dataBits) {case 5: cfg.c_cflag |= CS5; break;case 6: cfg.c_cflag |= CS6; break;case 7: cfg.c_cflag |= CS7; break;case 8: cfg.c_cflag |= CS8; break;default: cfg.c_cflag |= CS8; break;}switch (stopBits) {case 1: cfg.c_cflag &= ~CSTOPB; break;case 2: cfg.c_cflag |= CSTOPB; break;default:cfg.c_cflag &= ~CSTOPB; break;}if (tcsetattr(fd, TCSANOW, &cfg)){LOGE("tcsetattr() failed");close(fd);/* TODO: throw an exception */return NULL;}}/* Create a corresponding file descriptor */{jclass cFileDescriptor = (*env)->FindClass(env, "java/io/FileDescriptor");jmethodID iFileDescriptor = (*env)->GetMethodID(env, cFileDescriptor, "<init>", "()V");jfieldID descriptorID = (*env)->GetFieldID(env, cFileDescriptor, "descriptor", "I");mFileDescriptor = (*env)->NewObject(env, cFileDescriptor, iFileDescriptor);(*env)->SetIntField(env, mFileDescriptor, descriptorID, (jint)fd);}return mFileDescriptor;
}
  1. 修改SerialPort.java文件,将原来的构造方法修改成下面的
/*** 串口配置*@param device 串口设备文件*@param baudrate 波特率*@param parity 奇偶校验,0 None, 1 Odd, 2 Even*@param dataBits 数据位,5 - 8*@param stopBits 停止位,1 或 2*/public SerialPort(File device, int baudrate, int parity, int dataBits, int stopBits) throws SecurityException, IOException {/* Check access permission */if (!device.canRead() || !device.canWrite()) {try {/* Missing read/write permission, trying to chmod the file */Process su;su = Runtime.getRuntime().exec("/system/bin/su");String cmd = "chmod 666 " + device.getAbsolutePath() + "\n"+ "exit\n";su.getOutputStream().write(cmd.getBytes());if ((su.waitFor() != 0) || !device.canRead()|| !device.canWrite()) {throw new SecurityException();}} catch (Exception e) {e.printStackTrace();throw new SecurityException();}}mFd = open(device.getAbsolutePath(), baudrate, parity, dataBits, stopBits);if (mFd == null) {Log.e(TAG, "native open returns null");throw new IOException();}mFileInputStream = new FileInputStream(mFd);mFileOutputStream = new FileOutputStream(mFd);}

还有JNI的open方法

private native static FileDescriptor open(String path, int baudrate, int parity, int dataBits, int stopBits);

如何使用呢?

我用的是Android Studio 3.5,在新建项目的时候,是看不到Include C++ Support这个选项的,但也无妨,无论是新建的项目还是已经现有的项目都可以按照下面的步骤完成配置。

  1. 检查ndk配置
    在这里插入图片描述

NDK(Native Develop Kit),Android NDK 是一套允许您使用原生代码语言(例如C和C++) 实现部分应用的工具集。在开发某些类型应用时,这有助于您重复使用C/C++语言编写的代码库。

如果还没有下载Android NDK的可以借助Android Studio下载或者网上找资源下载,比如官方网站https://developer.android.google.cn/ndk/downloads/。哪种方式方便快捷,就可以用哪种。
Android Studio找到Settings,或者使用快捷键Ctrl + Alt + S。搜索Android SDK,找到SDK Tools,最下面就是NDK的版本信息,勾选上之后,点击Apply,最后点OK。
NDK下载完成,按照上面的步骤配置好NDK路径。
在这里插入图片描述

  1. 将整个工程目录切换至Project视图,默认是Android视图。找到src/main,右键main文件夹,选择New,找到Folder,选择JNI Folder
    在这里插入图片描述

JNI(Java Native Interface),即Java本地接口,JNI是Java调用Native 语言的一种特性。通过JNI可以使得Java与C/C++机型交互。即可以在Java代码中调用C/C++等语言的代码或者在C/C++代码中调用Java代码。

将官方库android-serialport-api下载下来,找到jni文件夹,将里面所有的文件复制到上面新建的jni文件夹里。如果想要支持奇偶检验、数据位和停止位设定,按照我上面的代码修改就可以实现。
在这里插入图片描述

  1. 配置app/build.gradle
android {...externalNativeBuild {ndkBuild {path 'src/main/jni/Android.mk'}}
}

4.最后一步,在src/main/java根目录下,新建一个文件夹android_serialport_api,名字千万不要改哦,因为这个名字链接着这个api库,改变之后,java代码无法调用它,会报错的。
新建一个SerialPort类或者将官方原来的SerialPort.java文件复制过去,如果有需要,也可以将SerialPortFinder.java这个文件复制过去,这个是查找串口驱动文件路径。

public class SerialPort {private static final String TAG = "SerialPort";/** Do not remove or rename the field mFd: it is used by native method close();*/private FileDescriptor mFd;private FileInputStream mFileInputStream;private FileOutputStream mFileOutputStream;/*** 串口配置*@param device 串口设备文件*@param baudrate 波特率*@param parity 奇偶校验,0 None, 1 Odd, 2 Even*@param dataBits 数据位,5 - 8*@param stopBits 停止位,1 或 2*/public SerialPort(File device, int baudrate, int parity, int dataBits, int stopBits) throws SecurityException, IOException {/* Check access permission */if (!device.canRead() || !device.canWrite()) {try {/* Missing read/write permission, trying to chmod the file */Process su;su = Runtime.getRuntime().exec("/system/bin/su");String cmd = "chmod 666 " + device.getAbsolutePath() + "\n"+ "exit\n";su.getOutputStream().write(cmd.getBytes());if ((su.waitFor() != 0) || !device.canRead()|| !device.canWrite()) {throw new SecurityException();}} catch (Exception e) {e.printStackTrace();throw new SecurityException();}}mFd = open(device.getAbsolutePath(), baudrate, parity, dataBits, stopBits);if (mFd == null) {Log.e(TAG, "native open returns null");throw new IOException();}mFileInputStream = new FileInputStream(mFd);mFileOutputStream = new FileOutputStream(mFd);}// Getters and setterspublic InputStream getInputStream() {return mFileInputStream;}public OutputStream getOutputStream() {return mFileOutputStream;}// JNIprivate native static FileDescriptor open(String path, int baudrate, int parity, int dataBits, int stopBits);public native void close();static {System.loadLibrary("serial_port");}
}

到这里,Google官方库android-serialport-api基本配置完成了,剩下的就是写工具类进行调用了,实现打开串口、监听数据、发送数据和关闭串口了。

如果不知道如何下手的,我下面写了SerialPortHelper,可以参考参考下。

public abstract class SerialPortHelper
{private SerialPort mSerialPort;private OutputStream mOutputStream;private InputStream mInputStream;private ReadThread mReadThread;private boolean _isOpen = false;//串口配置private String sPort;private int iBaudRate;private int parity = 0; //默认无校验private int dataBits = 8;//默认数据位8位private int stopBits = 1;//默认停止位1位public SerialPortHelper(String sPort, int iBaudRate) {this.sPort = sPort;this.iBaudRate = iBaudRate;}public SerialPortHelper(String sPort, int iBaudRate, int parity, int dataBits, int stopBits) {this.sPort = sPort;this.iBaudRate = iBaudRate;this.parity = parity;this.dataBits = dataBits;this.stopBits = stopBits;}public void open() throws SecurityException, IOException {this.mSerialPort = new SerialPort(new File(sPort), iBaudRate, parity, dataBits, stopBits);this.mOutputStream = this.mSerialPort.getOutputStream();this.mInputStream = this.mSerialPort.getInputStream();this.mReadThread = new ReadThread();this.mReadThread.start();this._isOpen = true;}public void close() {if (this.mReadThread != null) {this.mReadThread.interrupt();}if (this.mSerialPort != null) {this.mSerialPort.close();this.mSerialPort = null;}this._isOpen = false;}public void send(byte[] bOutArray) {try {this.mOutputStream.write(bOutArray);} catch (IOException e) {e.printStackTrace();}}public void sendHex(String sHex) {byte[] bOutArray = ByteUtil.HexToByteArr(sHex);send(bOutArray);}public void sendTxt(String sTxt) {byte[] bOutArray = sTxt.getBytes();send(bOutArray);}private class ReadThread extends Thread {@Overridepublic void run() {super.run();while (!isInterrupted()) {try {if (SerialPortHelper.this.mInputStream == null) {return;}int available = SerialPortHelper.this.mInputStream.available();if (available > 0) {byte[] buffer = new byte[available];int size = SerialPortHelper.this.mInputStream.read(buffer);if (size > 0) {SerialPortHelper.this.onDataReceived(buffer, size);}} else {SystemClock.sleep(50);}} catch (Throwable e) {Log.e("error", e.getMessage());return;}}}}public boolean isOpen() {return this._isOpen;}protected abstract void onDataReceived(byte[] buffer, int size);}

初始化串口


private SerialPortHelper mSerialPortHelper;mSerialPortHelper = new SerialPortHelper("dev/ttyS4", 9600) {@Overrideprotected void onDataReceived(byte[] buffer, int size) {//todo 业务处理,解析接收的数据}
};

打开串口

mSerialPortHelper.open();

发送数据

mSerialPortHelper.send(byte[]/Hex String/Txt String);

关闭串口

mSerialPortHelper.close();

如果觉得上面的步骤比较繁琐,开发效率不高的话,可以试试别人造的轮子。
快速使用Android串口

这篇关于Android串口使用2之使用Google官方库android-serialport-api的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/496208

相关文章

Python并行处理实战之如何使用ProcessPoolExecutor加速计算

《Python并行处理实战之如何使用ProcessPoolExecutor加速计算》Python提供了多种并行处理的方式,其中concurrent.futures模块的ProcessPoolExecu... 目录简介完整代码示例代码解释1. 导入必要的模块2. 定义处理函数3. 主函数4. 生成数字列表5.

Python中help()和dir()函数的使用

《Python中help()和dir()函数的使用》我们经常需要查看某个对象(如模块、类、函数等)的属性和方法,Python提供了两个内置函数help()和dir(),它们可以帮助我们快速了解代... 目录1. 引言2. help() 函数2.1 作用2.2 使用方法2.3 示例(1) 查看内置函数的帮助(

Linux脚本(shell)的使用方式

《Linux脚本(shell)的使用方式》:本文主要介绍Linux脚本(shell)的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录概述语法详解数学运算表达式Shell变量变量分类环境变量Shell内部变量自定义变量:定义、赋值自定义变量:引用、修改、删

Java使用HttpClient实现图片下载与本地保存功能

《Java使用HttpClient实现图片下载与本地保存功能》在当今数字化时代,网络资源的获取与处理已成为软件开发中的常见需求,其中,图片作为网络上最常见的资源之一,其下载与保存功能在许多应用场景中都... 目录引言一、Apache HttpClient简介二、技术栈与环境准备三、实现图片下载与保存功能1.

Python中使用uv创建环境及原理举例详解

《Python中使用uv创建环境及原理举例详解》uv是Astral团队开发的高性能Python工具,整合包管理、虚拟环境、Python版本控制等功能,:本文主要介绍Python中使用uv创建环境及... 目录一、uv工具简介核心特点:二、安装uv1. 通过pip安装2. 通过脚本安装验证安装:配置镜像源(可

LiteFlow轻量级工作流引擎使用示例详解

《LiteFlow轻量级工作流引擎使用示例详解》:本文主要介绍LiteFlow是一个灵活、简洁且轻量的工作流引擎,适合用于中小型项目和微服务架构中的流程编排,本文给大家介绍LiteFlow轻量级工... 目录1. LiteFlow 主要特点2. 工作流定义方式3. LiteFlow 流程示例4. LiteF

使用Python开发一个现代化屏幕取色器

《使用Python开发一个现代化屏幕取色器》在UI设计、网页开发等场景中,颜色拾取是高频需求,:本文主要介绍如何使用Python开发一个现代化屏幕取色器,有需要的小伙伴可以参考一下... 目录一、项目概述二、核心功能解析2.1 实时颜色追踪2.2 智能颜色显示三、效果展示四、实现步骤详解4.1 环境配置4.

使用jenv工具管理多个JDK版本的方法步骤

《使用jenv工具管理多个JDK版本的方法步骤》jenv是一个开源的Java环境管理工具,旨在帮助开发者在同一台机器上轻松管理和切换多个Java版本,:本文主要介绍使用jenv工具管理多个JD... 目录一、jenv到底是干啥的?二、jenv的核心功能(一)管理多个Java版本(二)支持插件扩展(三)环境隔

SQL中JOIN操作的条件使用总结与实践

《SQL中JOIN操作的条件使用总结与实践》在SQL查询中,JOIN操作是多表关联的核心工具,本文将从原理,场景和最佳实践三个方面总结JOIN条件的使用规则,希望可以帮助开发者精准控制查询逻辑... 目录一、ON与WHERE的本质区别二、场景化条件使用规则三、最佳实践建议1.优先使用ON条件2.WHERE用

Java中Map.Entry()含义及方法使用代码

《Java中Map.Entry()含义及方法使用代码》:本文主要介绍Java中Map.Entry()含义及方法使用的相关资料,Map.Entry是Java中Map的静态内部接口,用于表示键值对,其... 目录前言 Map.Entry作用核心方法常见使用场景1. 遍历 Map 的所有键值对2. 直接修改 Ma