开发串口程序首先要求你的设备需要支持串口通信,可以在设备上装一个App端的串口工具来检测一下

链接:https://pan.baidu.com/s/11L4aZI9orBhbnztka6H1Og 
提取码:bvot 

或者在电脑端下载一个友善串口助手检测一下,一般在Android工控主板上面都会带有串口。

首先我们是用到了谷歌开源的API serialPort 

先贴出来下载地址 https://github.com/cepr/android-serialport-api

第一步  配置环境

1、开发工具Android studio,2.2-3.1.2都可以

2、配置NDK(http://blog.csdn.net/yehui928186846/article/details/52787773),网上教程很多,这里不做重点讲解,查看配置是否成功

 

 

 

 

 

 

 

3、Android studio配置ndk

 

 

 

 

 

 

 

二、用开源库代码复制到自己项目里

1、如下图所示

里面的操作类我做了重构,可能会跟开源里面的不一样,不过都是为了达到自己的需求嘛

2、配置build-gridle

配置信息直接粘上去就可以了

在project目录下的gradle.properties文件内加上

Android.useDeprecatedNdk=true这句话 ,为了兼容新老版本ndk

3、类的讲解

public class SerialPortFinder {    public class Driver {        public Driver(String name, String root) {            mDriverName = name;            mDeviceRoot = root;        }        private String mDriverName;        private String mDeviceRoot;        Vector mDevices = null;        public Vector getDevices() {            if (mDevices == null) {                mDevices = new Vector();                File dev = new File("/dev");                File[] files = dev.listFiles();                int i;                for (i = 0; i < files.length; i++) {                    if (files[i].getAbsolutePath().startsWith(mDeviceRoot)) {                        Log.d(TAG, "Found new device: " + files[i]);                        mDevices.add(files[i]);                    }                }            }            return mDevices;        }        public String getName() {            return mDriverName;        }    }    private static final String TAG = "SerialPort";    private Vector mDrivers = null;    Vector getDrivers() throws IOException {        if (mDrivers == null) {            mDrivers = new Vector();            LineNumberReader r = new LineNumberReader(new FileReader("/proc/tty/drivers"));            String l;            while ((l = r.readLine()) != null) {                // Issue 3:                // Since driver name may contain spaces, we do not extract driver name with split()                String drivername = l.substring(0, 0x15).trim();                String[] w = l.split(" +");                if ((w.length >= 5) && (w[w.length - 1].equals("serial"))) {                    Log.d(TAG, "Found new driver " + drivername + " on " + w[w.length - 4]);                    mDrivers.add(new Driver(drivername, w[w.length - 4]));                }            }            r.close();        }        return mDrivers;    }    public String[] getAllDevices() {        Vector devices = new Vector();        // Parse each driver        Iterator itdriv;        try {            itdriv = getDrivers().iterator();            while (itdriv.hasNext()) {                Driver driver = itdriv.next();                Iterator itdev = driver.getDevices().iterator();                while (itdev.hasNext()) {                    String device = itdev.next().getName();                    String value = String.format("%s (%s)", device, driver.getName());                    devices.add(value);                }            }        } catch (IOException e) {            e.printStackTrace();        }        return devices.toArray(new String[devices.size()]);    }//获取设备上所有的串口节点    public String[] getAllDevicesPath() {        Vector devices = new Vector();        // Parse each driver        Iterator itdriv;        try {            itdriv = getDrivers().iterator();            while (itdriv.hasNext()) {                Driver driver = itdriv.next();                Iterator itdev = driver.getDevices().iterator();                while (itdev.hasNext()) {                    String device = itdev.next().getAbsolutePath();                    devices.add(device);                }            }        } catch (IOException e) {            e.printStackTrace();        }        return devices.toArray(new String[devices.size()]);    }}

这个类一般不用,不占主要作用,主要用于可以获取设备上的所有可用的串口节点,用来选择设置,根据需求添加

 

public class SerialPort {    private static final String TAG = "SerialPort";    private FileDescriptor mFd;    private FileInputStream mFileInputStream;    private FileOutputStream mFileOutputStream;    public SerialPort(File device, int baudrate, int flags) throws SecurityException, IOException {        //检查访问权限,如果没有读写权限,进行文件操作,修改文件访问权限        if (!device.canRead() || !device.canWrite()) {            try {                //通过挂载到linux的方式,修改文件的操作权限                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.printStackTrace();                throw new SecurityException();            }        }        mFd = open(device.getAbsolutePath(), baudrate, flags);        if (mFd == null) {            Log.e(TAG, "native open returns null");            throw new IOException();        }        mFileInputStream = new FileInputStream(mFd);        mFileOutputStream = new FileOutputStream(mFd);    }    // Getters and setters    public InputStream getInputStream() {        return mFileInputStream;    }    public OutputStream getOutputStream() {        return mFileOutputStream;    }    // JNI(调用java本地接口,实现串口的打开和关闭)    /**     * 串口有五个重要的参数:串口设备名,波特率,检验位,数据位,停止位     * 其中检验位一般默认位NONE,数据位一般默认为8,停止位默认为1     */    /**     * @param path     串口设备的绝对路径     * @param baudrate 波特率     * @param flags    校验位     */    private native static FileDescriptor open(String path, int baudrate, int flags);    public native void close();    static {//加载jni下的C文件库        System.loadLibrary("serial_port");    }}

这个SerialPort类是开源的,没有经过修改,Android可以,里面的直接调用,native方法直接和C通信,我们做Android的不需要管

jni目录下放着c源码和h头文件,

jniLibs下面放的就是so库。

注意:因为用的谷歌原生so库,所以SerialPort类的包名一定要是android_serialport_api,如果想修改这个包名,就需要重新生成对应的so库

 

public class SerialPortUtil {    public static String TAG = "SerialPortUtil";    /**     * 标记当前串口状态(true:打开,false:关闭)     **/    public static boolean isFlagSerial = false;    public static SerialPort serialPort = null;    public static InputStream inputStream = null;    public static OutputStream outputStream = null;    public static Thread receiveThread = null;    public static String strData = "";    public static Handler mHandler;    /**     * 打开串口     */    public static boolean open() {        boolean isopen = false;        if(isFlagSerial){            LogUtils.e(TAG,"串口已经打开,打开失败");            return false;        }        try {            serialPort = new SerialPort(new File("/dev/ttyS3"), 115200, 0);            inputStream = serialPort.getInputStream();            outputStream = serialPort.getOutputStream();            receive();            isopen = true;            isFlagSerial = true;        } catch (IOException e) {            e.printStackTrace();            isopen = false;        }        return isopen;    }    /**     * 关闭串口     */    public static boolean close() {        if(isFlagSerial){            LogUtils.e(TAG,"串口关闭失败");            return false;        }        boolean isClose = false;        LogUtils.e(TAG, "关闭串口");        try {            if (inputStream != null) {                inputStream.close();            }            if (outputStream != null) {                outputStream.close();            }            isClose = true;            isFlagSerial = false;//关闭串口时,连接状态标记为false        } catch (IOException e) {            e.printStackTrace();            isClose = false;        }        return isClose;    }    /**     * 发送串口指令     */    public static void sendString(String data, Handler handler) {        mHandler = handler;        if (!isFlagSerial) {            LogUtils.e(TAG, "串口未打开,发送失败" + data);            return;        }        try {            outputStream.write(ByteUtil.hex2byte(data));            outputStream.flush();            LogUtils.e(TAG, "sendSerialData:" + data);        } catch (IOException e) {            e.printStackTrace();            LogUtils.e(TAG, "发送指令出现异常");        }    }    /**     * 接收串口数据的方法     */    public static void receive() {        if (receiveThread != null && !isFlagSerial) {            return;        }        receiveThread = new Thread() {            @Override            public void run() {                while (isFlagSerial) {                    try {                        byte[] readData = new byte[32];                        if (inputStream == null) {                            return;                        }                        int size = inputStream.read(readData);                        if (size > 0 && isFlagSerial) {                            strData = ByteUtil.byteToStr(readData, size);                            LogUtils.e(TAG, "readSerialData:" + strData);                        }                    } catch (IOException e) {                        e.printStackTrace();                    }                }            }        };        receiveThread.start();    }}

这个类就比较重要了,打开串口、关闭串口、读写操作,都在这个类里面写了详细的注释,另外下面在贴一个工具类出来

package com.sqy.scancode.util;import android.graphics.Bitmap;import android.graphics.BitmapFactory;import android.util.Base64;import java.io.FileInputStream;import java.io.IOException;import java.io.InputStream;import Decoder.BASE64Decoder;import Decoder.BASE64Encoder;/** * Created by Administrator on 2018/6/15. */public class ByteUtil {    /**     * 字符串转化成为16进制字符串     *     * @param s     * @return     */    public static String strTo16(String s) {        String str = "";        for (int i = 0; i < s.length(); i++) {            int ch = (int) s.charAt(i);            String s4 = Integer.toHexString(ch);            str = str + s4;        }        return str;    }    /**     * 16进制转换成为string类型字符串     *     * @param s     * @return     */    public static String hexStringToString(String s) {        if (s == null || s.equals("")) {            return null;        }        s = s.replace(" ", "");        byte[] baKeyword = new byte[s.length() / 2];        for (int i = 0; i < baKeyword.length; i++) {            try {                baKeyword[i] = (byte) (0xff & Integer.parseInt(s.substring(i * 2, i * 2 + 2), 16));            } catch (Exception e) {                e.printStackTrace();            }        }        try {            s = new String(baKeyword, "UTF-8");            new String();        } catch (Exception e1) {            e1.printStackTrace();        }        return s;    }    /**     * 向串口发送数据转为字节数组     */    public static byte[] hex2byte(String hex) {        String digital = "0123456789ABCDEF";        String hex1 = hex.replace(" ", "");        char[] hex2char = hex1.toCharArray();        byte[] bytes = new byte[hex1.length() / 2];        byte temp;        for (int p = 0; p < bytes.length; p++) {            temp = (byte) (digital.indexOf(hex2char[2 * p]) * 16);            temp += digital.indexOf(hex2char[2 * p + 1]);            bytes[p] = (byte) (temp & 0xff);        }        return bytes;    }    /**     * 接收到的字节数组转换16进制字符串     */    public static String bytes2HexString(byte[] b, int size) {        String ret = "";        for (int i = 0; i < size; i++) {            String hex = Integer.toHexString(b[i] & 0xFF);            if (hex.length() == 1) {                hex = '0' + hex;            }            ret += hex.toUpperCase();        }        return ret;    }    public static String bytesToHexString(byte[] src) {        StringBuilder stringBuilder = new StringBuilder("");        if (src == null || src.length <= 0) {            return null;        }        for (int i = 0; i < src.length; i++) {            int v = src[i] & 0xFF;            String hv = Integer.toHexString(v);            if (hv.length() < 2) {                stringBuilder.append(0);            }            stringBuilder.append(hv);        }        return stringBuilder.toString();    }    /**     * 接收到的字节数组转换16进制字符串     */    public static String byteToStr(byte[] b, int size) {        String ret = "";        for (int i = 0; i < size; i++) {            String hex = Integer.toHexString(b[i] & 0xFF);            if (hex.length() == 1) {                hex = '0' + hex;            }            ret += hex.toUpperCase();        }        return ret;    }    /**     * BASE64码解密成图片     */    public static Bitmap Base64ToImage(String imgStr) { // 对字节数组字符串进行Base64解码并生成图片        BASE64Decoder decoder = new BASE64Decoder();        Bitmap bitmap = null;        try {            // Base64解码            byte[] b = decoder.decodeBuffer(imgStr);            for (int i = 0; i < b.length; ++i) {                if (b[i] < 0) {// 调整异常数据                    b[i] += 256;                }            }            bitmap = BitmapFactory.decodeByteArray(b,0,b.length);            return bitmap;        } catch (Exception e) {            LogUtils.e("TAG","解析异常");            return bitmap;        }    }    /**     * 将图片转换为base64加密数据     */    public static String ImageToBase64(String imgFile) {        InputStream in = null;        byte[] data = null;        try {            in = new FileInputStream(imgFile);            data = new byte[in.available()];            in.read(data);            in.close();        } catch (IOException e) {            LogUtils.e("TAG","加密异常");            e.printStackTrace();        }        BASE64Encoder encoder = new BASE64Encoder();        return encoder.encode(data);    }    /**     * 计算CRC16校验码     * 逐个求和     *     * @param bytes 字节数组     * @return {@link String} 校验码     * @since 1.0     */    public static String getCRC_16(byte[] bytes) {        int CRC = 0x0000ffff;        int POLYNOMIAL = 0x0000a001;        int i, j;        for (i = 0; i < bytes.length; i++) {            CRC ^= ((int) bytes[i] & 0x000000ff);            for (j = 0; j < 8; j++) {                if ((CRC & 0x00000001) != 0) {                    CRC >>= 1;                    CRC ^= POLYNOMIAL;                } else {                    CRC >>= 1;                }            }        }        if (Integer.toHexString(CRC).toUpperCase().length() == 2) {            return byteToStr(bytes, bytes.length) + "00" + Integer.toHexString(CRC).toUpperCase();        } else if (Integer.toHexString(CRC).toUpperCase().length() == 3) {            return byteToStr(bytes, bytes.length) + "0" + Integer.toHexString(CRC).toUpperCase();        }        return byteToStr(bytes, bytes.length) + Integer.toHexString(CRC).toUpperCase();    }    /**     * 指令校验和,并取出后两位字节     * */    public static String getSum16(byte[] msg, int length) {        long mSum = 0;        byte[] mByte = new byte[length];        /** 逐Byte添加位数和 */        for (byte byteMsg : msg) {            long mNum = ((long) byteMsg >= 0) ? (long) byteMsg : ((long) byteMsg + 256);            mSum += mNum;        } /** end of for (byte byteMsg : msg) */        /** 位数和转化为Byte数组 */        for (int liv_Count = 0; liv_Count < length; liv_Count++) {            mByte[length - liv_Count - 1] = (byte) (mSum >> (liv_Count * 8) & 0xff);        } /** end of for (int liv_Count = 0; liv_Count < length; liv_Count++) */        return byteToStr(msg, length) + byteToStr(mByte, mByte.length).substring(byteToStr(mByte, mByte.length).length() - 4, byteToStr(mByte, mByte.length).length());    }}

4、demo下载地址 : https://github.com/z-jc/ScanCode   里面可能还会有一些别的功能,需要的话自行下载

5、另外再提供一个自己封装好的module,https://github.com/z-jc/SerialProject-master,app导入module

然后在activity内直接这样调用

省心又省劲是不是,用到串口的项目可以直接当一个libray导入项目,不过需要检验的话还是得根据自己的校验方法来进行校验

以上全为原创,如有讲解不到之处,还请广大朋友指点一下

下一篇 Android 串口开发(二) 支持设置奇偶校验、数据位、停止位

更多相关文章

  1. 利用Bundle实现Android(安卓)Activity间消息的传递
  2. Android中解析JSON(一)
  3. 【Android】Replace "..." with ellipsis character (…, &&;#82
  4. Android(安卓)串口通讯 获取卡号
  5. 基于 Android(安卓)NDK 的学习之旅----- Java 方法映射到C中的签
  6. android 限制输入字符数!看清楚:是限制字符数,不是字数!中英混排?就算
  7. Android面试题目之三: 字符串转整形
  8. Android(安卓)ScrollView用法
  9. android 根据图片的URI获取对应的图片

随机推荐

  1. Android与H5相机、相册笔记
  2. JPCT-AE for Android(安卓)3D (一)------
  3. Android内存泄露调试分享
  4. Android(安卓)滚动条
  5. Android动态壁纸的制作教程
  6. Android学习笔记1——建立android工程及
  7. Android(安卓)技术专题系列之九 -- 图形系
  8. 英特尔® Android* USB 驱动程序安装指南
  9. Android开发实例详解之IMF(Android(安卓)S
  10. 如何自学 Android(安卓)编程?(励志)