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);
}
(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 := SerialPort
LOCAL_C_INCLUDES := $(LOCAL_PATH)/include
LOCAL_SRC_FILES := SerialPort.cpp
LOCAL_LDLIBS += -llog
include $(BUILD_SHARED_LIBRARY)
SerialPort.cpp
#include
#include
#include
#include
#include
#include
#include
#include
#include
#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, "", "()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 (这里填写与你生成的.h文件中的close方法名一致,括号不用)
(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;
}
}
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,则里面填写['src/main/libs']
}
}
}