Android串口开发之使用JNI实现ANDROID和串口通信详解

一:串口通信简介

前段时间因为工作需要研究了一下android的串口通信,网上有很多讲串口通信的文章,我在做的时候也参考了很多文章,现在就将我学习过程中的一些心得分享给大家,由于串口开发涉及到jni,所以开发环境需要支持ndk开发,如果未配置ndk配置的朋友,或者对jni不熟悉的朋友,请查看上一篇文章,android 串口开发第一篇:搭建ndk开发环境以及第一个jni调用程序 ,串口通信和java操作io类似,先打开串口,然后向串口发送或者读取数据,最后关闭串口,所以基本思路就是:

  1.对串口文件进行配置(波特率等),选择串口文件,打开串口,设备不同 ,可以读写的串口也不同.

  2.读写串口 ,读串口需要开一个子线程,然后死循环读取串口发送的数据

  3.关闭串口文件

其中打开,关闭串口是在jni方法执行,读写操作是android程序执行。

二:代码实现

我的开发环境是android studio 2.3.3 串口开发我创建一个支持c++项目,然后在cpp目录下,创建一个nateve-lib.cpp的程序,将串口打开,串口关闭的程序复制进去即可,native-lib程序中方法的命名规则需要根据你实际情况,稍作修改,cpp中方法名格式为,Java_包名_调用jni方法的类名_方法名,如Java_com_serialportdemo_SerialPort_open,此处一定要注意,android studio生成的是cpp程序,不是c程序,这两个有一些区别的,比如:

我对c也不熟悉,以下语法有误请指出

*.c的语法

变量定义

jstring jstr2 = (*env) -> NewStringUTF(env, cstr);

方法定义

JNIEXPORT jstring JNICALL Java_com_serialportdemo_MainActivity_encode()
JNIEXPORT jstring JNICALL Java_com_serialportdemo_MainActivity_decode()

*.cpp的语法

jstring jstr2 =env->NewStringUTF(hello.c_str());

extern "C" //如果这里不写extern "C",程序编译不会错,但android无法调用该方法,错误日志是找不到该方法
JNIEXPORT jstring JNICALL Java_com_serialportdemo_MainActivity_encode()

extern "C"
JNIEXPORT jstring JNICALL Java_com_serialportdemo_MainActivity_decode()

串口打开,串口关闭代码如下:

//获取波特率
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;
 }
}
//打开串口程序
extern "C"
JNIEXPORT jobject JNICALL
Java_com_serialportdemo_SerialPort_open(JNIEnv *env, jobject thiz, jstring path,jint baudrate) {
 int fd;
 speed_t speed;
 jobject mFileDescriptor;
 LOGD("init native Check arguments");
 /* Check arguments */
 {
 speed = getBaudrate(baudrate);
 if (speed == -1) {
 /* TODO: throw an exception */
 LOGE("Invalid baudrate");
 return NULL;
 }
 }
 LOGD("init native Opening device!");
 /* Opening device */
 {
 jboolean iscopy;
 const char *path_utf = env->GetStringUTFChars(path, &iscopy);
 LOGD("Opening serial port %s", path_utf);
// fd = open(path_utf, O_RDWR | O_DIRECT | O_SYNC);
 fd = open(path_utf, O_RDWR | O_NOCTTY | O_NONBLOCK | O_NDELAY);
 LOGD("open() fd = %d", fd);
 env->ReleaseStringUTFChars(path, path_utf);
 if (fd == -1) {
 /* Throw an exception */
 LOGE("Cannot open port %d",baudrate);
 /* TODO: throw an exception */
 return NULL;
 }
 }
 LOGD("init native Configure device!");
 /* Configure device */
 {
 struct termios cfg;
 if (tcgetattr(fd, &cfg)) {
 LOGE("Configure device tcgetattr() failed 1");
 close(fd);
 return NULL;
 }
 cfmakeraw(&cfg);
 cfsetispeed(&cfg, speed);
 cfsetospeed(&cfg, speed);
 if (tcsetattr(fd, TCSANOW, &cfg)) {
 LOGE("Configure device tcsetattr() failed 2");
 close(fd);
 /* TODO: throw an exception */
 return NULL;
 }
 }
 /* Create a corresponding file descriptor */
 {
 jclass cFileDescriptor = env->FindClass("java/io/FileDescriptor");
 jmethodID iFileDescriptor = env->GetMethodID(cFileDescriptor,"<init>", "()V");
 jfieldID descriptorID = env->GetFieldID(cFileDescriptor,"descriptor", "I");
 mFileDescriptor = env->NewObject(cFileDescriptor,iFileDescriptor);
 env->SetIntField(mFileDescriptor, descriptorID, (jint) fd);
 }
 return mFileDescriptor;
}
//关闭串口程序
 extern "C"
JNIEXPORT jint JNICALL
Java_com_serialportdemo_SerialPort_close(JNIEnv * env, jobject thiz)
{
 jclass SerialPortClass = env->GetObjectClass(thiz);
 jclass FileDescriptorClass = env->FindClass("java/io/FileDescriptor");
 jfieldID mFdID = env->GetFieldID(SerialPortClass, "mFd", "Ljava/io/FileDescriptor;");
 jfieldID descriptorID = env->GetFieldID(FileDescriptorClass, "descriptor", "I");
 jobject mFd = env->GetObjectField(thiz, mFdID);
 jint descriptor = env->GetIntField(mFd, descriptorID);
 LOGD("close(fd = %d)", descriptor);
 close(descriptor);
 return 1;
}

android 方法就简单多了,首先来看串口操作类,在这个类中打开串口,测试没有做关闭串口的操作,jni的open方法,返回一个java.io.FileDescriptor对像,串口操作类通过该对像,获取文件的读写流操作对像.

//加载so文件
 static {
 System.loadLibrary("native-lib");
 }
/**
 * @param path 串口文件路径
 * @param baudrate 波特率,不同设备波特率有区别
 * */
 public SerialPort(String path, int baudrate) throws SecurityException, IOException {
 File device = new File(path);
 Logger.d(serialPortMsg());
 if(!device.canRead() || !device.canWrite()) {
 try {
 Process su = Runtime.getRuntime().exec("/system/bin/su");
 String cmd = "chmod 777 " + 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.getMessage();
 }
 }
 mFd = open(device.getAbsolutePath(), baudrate);
 Logger.d(TAG+"open commplete");
 if (mFd == null) {
 Logger.e(TAG, "native open returns null");
 throw new IOException();
 }
 mFileInputStream = new FileInputStream(mFd);
 mFileOutputStream = new FileOutputStream(mFd);
 }
 //定义本地方法
public native FileDescriptor open(String path, int baudrate);
public native void close();

接下来需要定义一个读取串口信息的线程,用于获取串口发送给android的信息

class ReadSerialPortMsgThread implements Runnable{
 @Override
 public void run() {
  int size;
  byte buff[] = new byte[1024];
  final SimpleDateFormat sdf = new SimpleDateFormat("HH:mm:ss");
  while (true){
  try {
   if(mInputStream==null){
   return;
   }
   size = mInputStream.read(buff);
   if(size<=0){
   continue;
   }
   final String message = new String(buff,0,size);
   Logger.d(TAG+"接收到串口回调 "+message);
   seriapPortMsg.append(message);
   if(buff[size - 1] == '\n'){
   log.post(new Runnable() {
    @Override
    public void run() {
    log.setText(sdf.format(new Date())+"接收到串口发送的指令 "+message);
    }
   });
   }
  }catch (Exception e){
   e.printStackTrace();
  }finally {
   try {
   Thread.sleep(1000);
   } catch (InterruptedException e) {
   e.printStackTrace();
   }
  }
  }
 }
 }

以上代码完成了对串口的读操作,串口写操作比较简单,就是得到串口的OutputStream,然后调用writer方法即可,代码如下:

@Override
 public void onClick(View view) {
 switch (view.getId()){
  case R.id.sendMsg:
  String msg = serMsg.getText().toString()+"\r\n";
  if(msg!=null&&!msg.equals("")){
   byte [] buff = msg.getBytes();
   try {
   mOutputStream.write(buff,0,buff.length);
   Logger.d(TAG+"msg 输出完成");
   } catch (IOException e) {
   e.printStackTrace();
   Logger.e(TAG+e.getMessage());
   }
  }
 }
 }

到此为止,读写操作的代码全部完成,我的测试串口设备一直在向android发送信息,如下图

三:注意事项

String SERIALPORT_NO3 = "/dev/ttyS3",int BAUDRATE=115200;  这是我设备定义的串口文件路径和波特率,这个信息位置需要根据实际情况作修改。

完整demo代码:https://github.com/jlq023/serialport (本地下载)

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • android串口开发入门之搭建ndk开发环境及第一个jni调用程序

    前言 这几天专门研究了下JNI编程,在网上找了好多资料,不过好多都是以前的,没有更新,而且有的还是错误的,让人不得不吐槽一把.所以觉得自己来一篇,本文将详细介绍关于android搭建ndk开发环境及第一个jni调用程序的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧. 一:ndk环境搭建 1:开发环境 我使用的是android studio 2.3.3版本,搭建ndk开发环境比较简单,打开File----Settings----Appearance&Behavior--

  • 详解Android JNI的基本使用(CMake)

    简介 什么是JNI JNI的全称是Java Native Interface:Java本地开发接口,它提供了若干的API实现了Java和其他语言的通信(主要是C和C++),目的就是Java可以调用C或C++开发的函数,C或C++也能调用Java的方法.这样有很多有点,其一就是效率,C/C++是本地语言,比java更高效:其二就是可以复用已经存在的C/C++代码:其三是Java反编译比C语言容易,一般加密算法都是用C语言编写,不容易被反编译. 什么是NDK和CMake NDK全称是Native D

  • Android快速开发系列 10个常用工具类实例代码详解

    打开大家手上的项目,基本都会有一大批的辅助类,今天特此整理出10个基本每个项目中都会使用的工具类,用于快速开发~~在此感谢群里给我发项目中工具类的兄弟/姐妹~ 1.日志工具类L.java package com.zhy.utils; import android.util.Log; /** * Log统一管理类 * * * */ public class L { private L() { /* cannot be instantiated */ throw new UnsupportedOpe

  • Android串口开发之使用JNI实现ANDROID和串口通信详解

    一:串口通信简介 前段时间因为工作需要研究了一下android的串口通信,网上有很多讲串口通信的文章,我在做的时候也参考了很多文章,现在就将我学习过程中的一些心得分享给大家,由于串口开发涉及到jni,所以开发环境需要支持ndk开发,如果未配置ndk配置的朋友,或者对jni不熟悉的朋友,请查看上一篇文章,android 串口开发第一篇:搭建ndk开发环境以及第一个jni调用程序 ,串口通信和java操作io类似,先打开串口,然后向串口发送或者读取数据,最后关闭串口,所以基本思路就是: 1.对串口文

  • Android开发之机顶盒上gridview和ScrollView的使用详解

    最近在机顶盒上做一个gridview, 其焦点需要在item的子控件上,但gridview的焦点默认在item上,通过 android:descendantFocusability="afterDescendants" <ScrollView android:id="@+id/scroll_content" android:layout_width="1740.0px" android:layout_height="600.0px

  • Android开发之保存图片到相册的三种方法详解

    目录 方法一 方法二 方法三 有三种方法如下:三个方法都需要动态申请读写权限否则保存图片到相册也会失败 方法一 /** * 保存bitmap到本地 * * @param bitmap Bitmap */ public static void saveBitmap(Bitmap bitmap, String path) { String savePath; File filePic; if (Environment.getExternalStorageState().equals(Environm

  • Android xUtils更新到3.0后的基本使用规则详解

    说实话,对于xUtils,是我最近才用到的开发框架(也是刚接触),对于其功能不得不说,简化了很多的开发步骤,可以说是非常好的开发工具,但是其最近更新到3.0也没有解决加载自定义ImageView报错的问题. xUtils简介 xUtils 包含了很多实用的android工具. xUtils 支持大文件上传,更全面的http请求协议支持(10种谓词),拥有更加灵活的ORM,更多的事件注解支持且不受混淆影响... xUitls 最低兼容android 2.2 (api level 8) 我总是喜欢用

  • Android Studio 3.6安装全过程及AVD安装运行步骤详解

    1.安装JDK1.7以上版本,Android Studio集成开发环境 (1)安装并检查JDK1.8,如图1.1所示 图1.1 配置完成的验证 (2)开始安装Android Studio集成开发环境,安装允许以管理员模式运行,如图1.2所示 图1.2 安装界面 (3)安装位置为F:\Studio\ android-studio-ide-192.6200805-windows,进行安装,如图1.3所示 图1.3 安装位置 (4)Android Studio集成开发环境安装完成,如图1.4所示 图1

  • Android 实现抖音小游戏潜艇大挑战的思路详解

    <潜水艇大挑战>是抖音上的一款小游戏,以面部识别来驱动潜艇通过障碍物,最近特别火爆,相信很多人都玩过. 一时兴起自己用Android自定义View也撸了一个,发现只要有好的创意,不用高深的技术照样可以开发出好玩的应用.开发过程现拿出来与大家分享一下. 项目地址: https://github.com/vitaviva/ugame 基本思路 整个游戏视图可以分成三层: camera(相机):处理相机的preview以及人脸识别 background(后景):处理障碍物相关逻辑 foregroun

  • Android 出现的警告(Service Intent must be explicit)解决办法详解

    Android 出现的警告(Service Intent must be explicit)解决办法详解 有些时候我们使用Service的时需要采用隐私启动的方式,但是Android 5.0一出来后,其中有个特性就是Service Intent  must be explitict,也就是说从Lollipop开始,service服务必须采用显示方式启动. 而android源码是这样写的(源码位置:sdk/sources/android-21/android/app/ContextImpl.jav

  • Android隐藏标题栏及解决启动闪过标题的实例详解

    Android隐藏标题栏及解决启动闪过标题的实例详解 方法一: 在代码中设置 this.requestWindowFeature(Window.FEATURE_NO_TITLE);//去掉标题栏 方法二: 在AndroidManifest.xml 里面设置 <application android:icon="@drawable/icon" android:label="@string/app_name" Android:theme="@androi

  • Android判断后台服务是否开启的两种方法实例详解

    Android判断后台服务是否开启的两种方法实例详解 最近项目用到后台上传,就开启了一个服务service. 但是刚开始用这种方法,有些机型不支持:酷派不支持.然后又换了第二种判断方法. // public boolean isServiceWork(Context mContext, String serviceName) { // boolean isWork = false; // ActivityManager myAM = (ActivityManager) mContext // .

随机推荐