android studio 串口通信JNI、NDK配置

来源:互联网 发布:网络安全法的特征是 编辑:程序博客网 时间:2024/06/06 06:36

android studio 串口通信JNI、NDK配置

最近刚好要做关于串口通信的项目,需要用到JNI,于是就去百度关于android这一块串口通信怎么使用,但是发现很多的配置都是eclipse的而关于studio的jni配置就很零散,另外,也是第一次弄,所以遇到了很多的问题,现在就把配置的流程写出来,方便以后如果不记得了,可以查看,当然,如果有遇到同样问题的人也可以通过这篇博客能够得到帮助。

1、先确认android studio中是否有NDK,如果没有可以通过studio下载相应的ndk版本,如下图所示:


点击之后,如果有ndk,则选择ndk路径,如果没有ndk,则直接点击下载ndk。


还有另外一种就是点击SDK manager,选择ndk下载即可


在local.properties文件中添加ndk的路径,不过这个一般studio更新操作的时候会自动加上去,如果不行,那就手动添加ndk的路径

ndk.dir=D\:\\Java\\sdk\\sdk\\android-ndk-r13-windows-x86_64\\android-ndk-r13
     在gradle.properties文件中添加这句话
android.useDeprecatedNdk=true
 2、在你的项目中新建文件夹放入SerialPort.java文件
** * 串口类,用于打开,关闭和控制串口的类.会用到JNI层的动态库. * * @author Hong */public class SerialPort {/** * 连接本地JNI动态库. */static {System.loadLibrary("SerialPort");}private static final String TAG = "SerialPort";private String deviceName;private int baudRate;private FileDescriptor fd;private FileInputStream fis;private FileOutputStream fos;/** * 构造函数 * * @param deviceName 串口设备名 * @param baudRate 串口波特率 */public SerialPort(String deviceName, int baudRate) {this.deviceName = deviceName;this.baudRate = baudRate;}/** * 打开串口设备.由于android没有串口相关的API,所以串口操作会调用本地 * JNI层的函数. * * @return 打开成功返回true,否则返回false. */public synchronized boolean openDevice() {if(deviceName == null || baudRate < 0)return false;fd = open(deviceName, baudRate);if(fd != null) {fis = new FileInputStream(fd);fos = new FileOutputStream(fd);return true;}return false;}/** * 关闭串口设备.会调用本地JNI层函数. * 同时关闭输入输出流 */public synchronized void closeDevice() {if(fd != null) {try {fos.close();fis.close();fos = null;fis = null;} catch (IOException e) {e.printStackTrace();}close(fd);fd = null;}}/** * 读串口数据 * * @param data 数据缓存 * @return 读取到的数据的实际大小 * @throws IOException */public int read(byte[] data) throws IOException {if(isSerialOpened()) {return fis.read(data);}return 0;}/** * 写入串口数据 * * @param data 要写入的数据 * @param offset 数据偏移量 * @param count 数据大小 * @throws IOException */public void write(byte[] data, int offset, int count) throws IOException {if(isSerialOpened()) {fos.write(data, offset, count);fos.flush();}}/** * 判断串口是否已经打开. * * @return 打开返回true,否则返回false. */public boolean isSerialOpened() {return fd != null;}/** * 获取串口设备名 * * @return 串口设备名 */public String getDeviceName() {return deviceName;}/** * 设置串口设备名.只有在串口还没被打开的情况下才能设置成功. * * @param deviceName 串口设备名 */public void setDeviceName(String deviceName) {if(!isSerialOpened())this.deviceName = deviceName;}/** * 获取串口波特率 * * @return 波特率 */public int getBaudRate() {return baudRate;}/** * 设置串口波特率.中有在串口还没被打开的情况下才能设置成功. * * @param baudRate */public void setBaudRate(int baudRate) {if(!isSerialOpened())this.baudRate = baudRate;}/** * 本地JNI层函数.打开串口设备.该函数的定义主体在 jni/serial_port.c文件中. * * @param path 要打开的串口设备的绝对路径. * @param baudrate 串口设备的波特率. * @return 成功打开返回串口设备的{@link FileDescriptor}实例.否则返回null. */private native FileDescriptor open(String path, int baudrate);/** * 本地JNI层函数.关闭串口设备.该函数的定义主体在 jni/serial_port.c文件中. * * @param fd 需要关闭的串口设备的{@link FileDescriptor}实例. * @return 成功关闭返回true,否则返回false. */private native boolean close(FileDescriptor fd);}

接下来,我们要根据这个文件生成.h文件,操作如下
(1)右击View->tool windows->terminal
在该terminal中,使用cd命令进入到项目的java目录下,输入javah -jni 包名.SerialPort,回车即可。在java的目录下就会生成项
目相应的.h文件,将该.h文件拷贝到jni文件夹下,jni文件目录下还有一个Android.mk和SerialPort.cpp文件,
Android.mk
LOCAL_PATH:= $(call my-dir)include $(CLEAR_VARS)LOCAL_MODULE := SerialPortLOCAL_C_INCLUDES := $(LOCAL_PATH)/includeLOCAL_SRC_FILES := SerialPort.cppLOCAL_LDLIBS += -lloginclude $(BUILD_SHARED_LIBRARY)


SerialPort.cpp
#include <stdio.h>#include <stdlib.h>#include <unistd.h>#include <sys/types.h>#include <sys/stat.h>#include <fcntl.h>#include <termios.h>#include <errno.h>#include <android/log.h>#include "你生成的.h文件,如xxx.h"#define LOGI(...)  __android_log_print(ANDROID_LOG_INFO, "SerialPort_JNI", "[JiuGui] "__VA_ARGS__)#define LOGE(...)  __android_log_print(ANDROID_LOG_ERROR, "SerialPort_JNI", "[JiuGui] "__VA_ARGS__)static speed_t getBaudRate(jint baudRate);/** * 与SerialPort.java中的open(String path, int baudRate)函数相关联。 * 作用是打开串口设备 */JNIEXPORT jobject JNICALL (这里填写与你生成的.h文件中的open方法名一致,括号不用)(JNIEnv *env, jobject object, jstring deviceName, jint baudRate){// 获取String的字符。不能直接的用device = deviceName;最后还要释放device指针。const char *device = env->GetStringUTFChars(deviceName, 0);LOGI("Device=%s, BaudRate= %d", device, baudRate);int fd = -1;do {// 打开设备fd = open(device, O_RDWR);if (-1 == fd) {LOGE("Can't Open %s[%d]: %s", device, errno, strerror(errno));break;}speed_t speed = getBaudRate(baudRate);if (speed == -1) {LOGE("Invalid baudRate");break;}//设置串口参数struct termios Opt;tcgetattr(fd, &Opt);cfmakeraw(&Opt);tcflush(fd, TCIFLUSH);cfsetispeed(&Opt, speed);cfsetospeed(&Opt, speed);if (tcsetattr(fd, TCSANOW, &Opt) != 0) {LOGE("SetupSerial![%d]: %s", errno, strerror(errno));break;}// 用JNI函数创建一个FileDescriptor类的实例jclass cFileDescriptor = env->FindClass("java/io/FileDescriptor");jmethodID iFileDescriptor = env->GetMethodID(cFileDescriptor, "<init>", "()V");jfieldID descriptorID = env->GetFieldID(cFileDescriptor, "descriptor", "I");jobject fileDescriptor = env->NewObject(cFileDescriptor, iFileDescriptor);env->SetIntField(fileDescriptor, descriptorID, (jint)fd);// 用完这个device指针后要释放。env->ReleaseStringUTFChars(deviceName, device);return fileDescriptor;} while(0);if(fd != -1)close(fd); //关闭设备// 用完这个device指针后要释放。env->ReleaseStringUTFChars(deviceName, device);return NULL;}/** * 与SerialPort.java中的close(FileDescriptor fd);函数相关联。 * 作用是关闭串口设备。 */JNIEXPORT jboolean JNICALL <span style="font-family: 宋体;">(这里填写与你生成的.h文件中的close方法名一致,括号不用)</span>(JNIEnv *env, jobject object, jobject descriptor){if(descriptor != NULL) {// 用JNI函数来得到参数descriptor里的“descriptor”属性值。// “descriptor”属性是在FileDescriptor类里的。jclass cFileDescriptor = env->FindClass("java/io/FileDescriptor");jfieldID descriptorID = env->GetFieldID(cFileDescriptor, "descriptor", "I");jint fd = env->GetIntField(descriptor, descriptorID);// 关闭设备。close(fd);}return 1;}/** * 获取波特率 */static speed_t getBaudRate(jint baudRate){switch(baudRate) {case 0: return B0;case 50: return B50;case 75: return B75;case 110: return B110;case 134: return B134;case 150: return B150;case 200: return B200;case 300: return B300;case 600: return B600;case 1200: return B1200;case 1800: return B1800;case 2400: return B2400;case 4800: return B4800;case 9600: return B9600;case 19200: return B19200;case 38400: return B38400;case 57600: return B57600;case 115200: return B115200;case 230400: return B230400;case 460800: return B460800;case 500000: return B500000;case 576000: return B576000;case 921600: return B921600;case 1000000: return B1000000;case 1152000: return B1152000;case 1500000: return B1500000;case 2000000: return B2000000;case 2500000: return B2500000;case 3000000: return B3000000;case 3500000: return B3500000;case 4000000: return B4000000;default: return -1;}}

将SerialPort.cpp中的添加头文件部分,以及相应的open、close方法对应于头文件中的open、close方法。
3、接下来就是要编译jni,生成相应的.so文件,在此过程中,我们需要配置一下ndk的环境,进入到ndk的文件目录下,拷贝ndk的
目录,打开电脑的系统设置-》高级系统设置-》环境变量,编辑path,将该路径粘贴到path中
点击确认。接着,打开命令行,使用cd命令进入到你的项目的jni目录下,输入ndk-build命令,点击回车,就会在项目的main文件
目录下生成libs和obj文件,libs中就会有.so文件,然后在项目的app的build.gradle文件中添加
 defaultConfig {        ndk {            abiFilter "armeabi"            moduleName "SerialPort"//你生成的.so文件的文件名            ldLibs "log", "z", "m", "jnigraphics", "android"        }    }
还需要添加
android {    sourceSets{ //设置.so文件路径        main{            jniLibs.srcDirs = ['libs']//里面添加你的libs文件路径,如果是src/main/libs,则里面填写['<span style="font-family: 宋体;">src/main/libs</span>']        }    }   }

至此,配置应该是算成功了。


0 0
原创粉丝点击