关于安卓串口编程的详解

来源:互联网 发布:java高级流和低级流 编辑:程序博客网 时间:2024/05/21 18:02
 由于近日在做关于用安卓手机端用串口配置调试底层设备,底层设备的通信接口为RS485通信方式,严格遵从modebus协议。此为大前提,如何通过安卓手机端完成这一工作,首先有两种方式:               1、OTG功能与底层连接;
     2、无需OTG功能;

    根据现在的安卓机中带有OTG功能手机所占的份额不是很大,因此,本项目选择无需OTG的通信方式。项目选择的设备为FT311D芯片的力特Z-TEK的android转RS232的设备为硬件通信设备。安卓端编程如下:首先将这个驱动芯片的类导入到工程
  1. <span style="background-color: rgb(255, 255, 255); ">public UsbManager usbmanager;</span>
复制代码
     这里有几个对象要注意下,这是usb通信中用到的对象,然后定义输入输出流,定义各种缓存数组,这里的maxnumbytes = 65536是最大字节数,是用于限制数据中最大不能超过2的16次方。下面注意的一点是在构造函数中要将USB通信挂起,这必须要在构造函数完成,先获取usb服务,用PendingIntent对象中的getBroadcast 的方法去注册我们定义的服务。然后判断usb是否连接,这里当usb设备拔出的时候会抛出一个action值为UsbManager.ACTION_USB_DEVICE_DETACHE,注册一个广播接收即可,同理当我们插入usb的时候也会自动的识别,然后输入输出流初始化为空。
  1. /*********************** USB 处理 ******************************************/
  2.                 usbmanager = (UsbManager) context.getSystemService(Context.USB_SERVICE);
  3.                 mPermissionIntent = PendingIntent.getBroadcast(context, 0, new Intent(
  4.                                 ACTION_USB_PERMISSION), 0);
  5.                 IntentFilter filter = new IntentFilter(ACTION_USB_PERMISSION);
  6.                 filter.addAction(UsbManager.ACTION_USB_ACCESSORY_DETACHED);
  7.                 context.registerReceiver(mUsbReceiver, filter);
  8.                 inputstream = null;
  9.                 outputstream = null;
复制代码
  1. </blockquote></div>   编写串口通信的基本配置函数,如下<div class="blockcode"><blockquote>public void SetConfig(int baud, byte dataBits, byte stopBits, byte parity,
  2.                         byte flowControl) {

  3.                 /* 准备波特率缓冲区 */
  4.                 writeusbdata[0] = (byte) baud;
  5.                 writeusbdata[1] = (byte) (baud >> 8);
  6.                 writeusbdata[2] = (byte) (baud >> 16);
  7.                 writeusbdata[3] = (byte) (baud >> 24);
  8.                 /* 数据位 */
  9.                 writeusbdata[4] = dataBits;
  10.                 /* 停止位 */
  11.                 writeusbdata[5] = stopBits;
  12.                 /* 校验位 */
  13.                 writeusbdata[6] = parity;
  14.                 /* 流控制 */
  15.                 writeusbdata[7] = flowControl;
  16.                 /* 发送UART配置包 */
  17.                 SendPacket((int) 8);
  18.                 Log.v(TGA, baud + "," + dataBits + "," + stopBits + "," + parity + ","
  19.                                 + flowControl);
  20.         }
复制代码
      这里的形参(int baud, byte dataBits, byte stopBits, byte parity,byte flowControl )分别代表串口的波特率、数据位、停止位、校验位、流控制。下面我简要说一下这几个参数:
1、波特率,代表通信过程中的传输速率,单位为bit/s,顾名思义也就是每秒钟传输多少位;
2、数据位,代表一次通信过程中传输的数据有多少个位(也可称一帧数据中数据位),一般分为两种7位或者8位,以8位用的最多;
3、停止位,就是我们所说的帧尾,这一位用于表示数据的截止,通信中一般用一位,但是对网络媒体通信中也可出现两位;
4、校验位,是将我们的一帧数据做校验,有无、奇校验、偶校验、MARK校验、SPACE校验几种;
5、流控制,指的是通信中的数据流,一般选择无即默认情况,但是在硬件的控制流中也会出现RTS/CTS流模式;

  1. /* 发送函数 */
  2.         public byte SendData(int numBytes, byte[] buffer) {
  3.                 /* 状态标志 */
  4.                 status = 0x00; /* 默认情况为成功 */
  5.                 /*
  6.                  * 如果超过最大限度num字节
  7.                  */
  8.                 if (numBytes < 1) {
  9.                         /* 返回的状态错误的命令 */
  10.                         return status;
  11.                 }

  12.                 /* 检查最大字节限制 */
  13.                 if (numBytes > 256) {
  14.                         numBytes = 256;
  15.                 }

  16.                 /* 准备要发送的数据包 */
  17.                 for (int count = 0; count < numBytes; count++) {
  18.                         writeusbdata[count] = buffer[count];
  19.                 }

  20.                 if (numBytes != 64) {
  21.                         SendPacket(numBytes);
  22.                 } else {
  23.                         byte temp = writeusbdata[63];
  24.                         SendPacket(63);
  25.                         writeusbdata[0] = temp;
  26.                         SendPacket(1);
  27.                 }

  28.                 return status;
  29.         }
复制代码
发送函数,这里就是用于数据的发送,故就不详细说明了。下面是接受函数也不做说明了,
  1. /* 读取数据 */
  2.         public byte ReadData(int numBytes, byte[] buffer, int[] actualNumBytes) {
  3.                 status = 0x00; /* 默认情况下成功 */

  4.                 /* 至少读取一个字节数据 */
  5.                 if ((numBytes < 1) || (totalBytes == 0)) {
  6.                         actualNumBytes[0] = 0;
  7.                         status = 0x01;
  8.                         return status;
  9.                 }

  10.                 /* 检查最大字节限制 */
  11.                 if (numBytes > totalBytes)
  12.                         numBytes = totalBytes;

  13.                 /* 更新可用的字节数 */
  14.                 totalBytes -= numBytes;

  15.                 actualNumBytes[0] = numBytes;

  16.                 /* 复制到用户缓冲区 */
  17.                 for (int count = 0; count < numBytes; count++) {
  18.                         buffer[count] = readBuffer[readIndex];
  19.                         readIndex++;
  20.                         /*
  21.                          * 拷贝到用户读取缓冲区,看是否超过什么 所以没有需要去检查溢出
  22.                          */
  23.                         readIndex %= maxnumbytes;
  24.                 }
  25.                 return status;
  26.         }
复制代码
     然后就是usb的广播函数,在前面我们注册的,这里的功能就是告诉安卓设备让不让我注册usb通信,让就在调试那里把允许调试打开,然后将自己电脑的调试指纹发送到手机中,我们会看到安卓设备会弹出一个界面。一堆指纹问你允不允许,这里想要通信的我建议允许,哈哈
  1. /*********** USB广播接收器 *******************************************/
  2.         private final BroadcastReceiver mUsbReceiver = new BroadcastReceiver() {
  3.                 @Override
  4.                 public void onReceive(Context context, Intent intent) {
  5.                         String action = intent.getAction();
  6.                         if (ACTION_USB_PERMISSION.equals(action)) {
  7.                                 synchronized (this) {
  8.                                         UsbAccessory accessory = (UsbAccessory) intent
  9.                                                         .getParcelableExtra(UsbManager.EXTRA_ACCESSORY);
  10.                                         if (intent.getBooleanExtra(
  11.                                                         UsbManager.EXTRA_PERMISSION_GRANTED, false)) {
  12.                                                 Toast.makeText(global_context, "允许USB许可",
  13.                                                                 Toast.LENGTH_SHORT).show();
  14.                                                 OpenAccessory(accessory);
  15.                                         } else {
  16.                                                 Toast.makeText(global_context, "否认USB许可",
  17.                                                                 Toast.LENGTH_SHORT).show();
  18.                                                 Log.d("LED", "permission denied for accessory "
  19.                                                                 + accessory);

  20.                                         }
  21.                                         mPermissionRequestPending = false;
  22.                                 }
  23.                         } else if (UsbManager.ACTION_USB_ACCESSORY_DETACHED.equals(action)) {
  24.                                 saveDetachPreference();
  25.                                 DestroyAccessory(true);
  26.                                 
  27.                         } else {
  28.                                 Log.d("LED", "....");
  29.                         }
  30.                 }
  31.         };
复制代码
然后就是我们的DestroyAccessory(true);函数就是注销函数如下所示:
  1. /* 注销 */
  2.         public void DestroyAccessory(boolean bConfiged) {

  3.                 if (true == bConfiged) {
  4.                         READ_ENABLE = false; // 设置错误条件handler_thread退出等待数据循环
  5.                         writeusbdata[0] = 0;
  6.                         SendPacket(1);
  7.                 } else {
  8.                         SetConfig(9600, (byte) 1, (byte) 8, (byte) 0, (byte) 0); // 发送默认设置数据配置

  9.                         try {
  10.                                 Thread.sleep(10);
  11.                         } catch (Exception e) {
  12.                         }

  13.                         READ_ENABLE = false; // 设置错误条件handler_thread退出等待数据循环
  14.                         writeusbdata[0] = 0; // send dummy data for instream.read going
  15.                         SendPacket(1);
  16.                         if (true == accessory_attached) {
  17.                                 saveDefaultPreference();
  18.                         }
  19.                 }

  20.                 try {
  21.                         Thread.sleep(10);
  22.                 } catch (Exception e) {
  23.                 }
  24.                 CloseAccessory();
  25.         }
复制代码
最后开辟一个线程用于处理usb通信
  1. /* usb输入数据处理程序 */
  2.         private class read_thread extends Thread {
  3.                 FileInputStream instream;

  4.                 read_thread(FileInputStream stream) {
  5.                         instream = stream;
  6.                         this.setPriority(Thread.MAX_PRIORITY);
  7.                 }

  8.                 public void run() {
  9.                         while (READ_ENABLE == true) {
  10.                                 while (totalBytes > (maxnumbytes - 1024)) {
  11.                                         try {
  12.                                                 Thread.sleep(50);
  13.                                         } catch (InterruptedException e) {
  14.                                                 e.printStackTrace();
  15.                                         }
  16.                                 }

  17.                                 try {
  18.                                         if (instream != null) {
  19.                                                 readcount = instream.read(usbdata, 0, 1024);
  20.                                                 if (readcount > 0) {
  21.                                                         for (int count = 0; count < readcount; count++) {
  22.                                                                 readBuffer[writeIndex] = usbdata[count];
  23.                                                                 writeIndex++;
  24.                                                                 writeIndex %= maxnumbytes;
  25.                                                         }

  26.                                                         if (writeIndex >= readIndex)
  27.                                                                 totalBytes = writeIndex - readIndex;
  28.                                                         else
  29.                                                                 totalBytes = (maxnumbytes - readIndex)
  30.                                                                                 + writeIndex;

  31.                                                         // Log.e(">>@@","totalBytes:"+totalBytes);
  32.                                                 }
  33.                                         }
  34.                                 } catch (IOException e) {
  35.                                         e.printStackTrace();
  36.                                 }
  37.                         }
  38.                 }
  39.         }
复制代码

     这个类先讲到这里,还有很多类,光它不会通信的(*^__^*) 嘻嘻……。下次我会继续讲解下面我们要在怎么做了,我会将它做成一个系列告诉大家一步一步的完成我现在做的这个项目后面更精彩哦!本人是开发者交流版块的版主看到排名下降,希望亲们感觉不错多多支持我们的版块哦。谢谢亲们  ~小轩,对了先附几张我软件的界面图吧,让大家了解一下~~~







Screenshot_2014-08-26-20-40-35.png (160.35 KB, 下载次数: 5)

Screenshot_2014-08-26-20-40-35.png

Screenshot_2014-08-26-20-40-40.png (668.04 KB, 下载次数: 2)

Screenshot_2014-08-26-20-40-40.png

Screenshot_2014-08-26-20-40-48.png (646.48 KB, 下载次数: 2)

Screenshot_2014-08-26-20-40-48.png

Screenshot_2014-08-26-20-40-53.png (421.29 KB, 下载次数: 2)

Screenshot_2014-08-26-20-40-53.png

Screenshot_2014-08-26-20-40-59.png (468.01 KB, 下载次数: 2)

Screenshot_2014-08-26-20-40-59.png

Screenshot_2014-08-26-20-41-05.png (69.17 KB, 下载次数: 2)

Screenshot_2014-08-26-20-41-05.png

Screenshot_2014-08-26-20-41-09.png (69.53 KB, 下载次数: 2)

Screenshot_2014-08-26-20-41-09.png

Screenshot_2014-08-26-20-41-21.png (149.86 KB, 下载次数: 2)

Screenshot_2014-08-26-20-41-21.png

Screenshot_2014-08-26-20-41-26.png (418.56 KB, 下载次数: 2)

Screenshot_2014-08-26-20-41-26.png

Screenshot_2014-08-26-20-41-33.png (62.74 KB, 下载次数: 2)

Screenshot_2014-08-26-20-41-33.png

Screenshot_2014-08-26-20-41-36.png (74.71 KB, 下载次数: 2)

Screenshot_2014-08-26-20-41-36.png

Screenshot_2014-08-26-20-41-42.png (74.19 KB, 下载次数: 0)

Screenshot_2014-08-26-20-41-42.png

Screenshot_2014-08-26-20-41-46.png (90.28 KB, 下载次数: 0)

Screenshot_2014-08-26-20-41-46.png

Screenshot_2014-08-28-14-25-30.png (164.18 KB, 下载次数: 0)

Screenshot_2014-08-28-14-25-30.png

Screenshot_2014-08-28-14-25-39.png (178.16 KB, 下载次数: 0)

Screenshot_2014-08-28-14-25-39.png

Screenshot_2014-08-28-14-26-01.png (675.04 KB, 下载次数: 0)

Screenshot_2014-08-28-14-26-01.png
0 0