Android AS下的OTG串口设备读写

2024-08-21 12:48
文章标签 android 串口 读写 设备 otg

本文主要是介绍Android AS下的OTG串口设备读写,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Android AS下的OTG串口设备读写

  • 新建工程
  • 添加工具类
  • 添加设备参数列表
  • 具体实现
  • WIFI adb

了解嵌入式的读者应该知道在单片机编程中串口(uart)通讯接口最常用的就是TTL和USB接口,将单片机TTL转USB就可以接入电脑查看串口数据实现电脑与单片机通讯,在Android AS下的NDK开发中讲解了Android使用TTL方式的接口收发数据,当然咱们常用的Android手机没有这样的接口,要实现手机和单片机串口通讯就可以用OTG来实现。

在这里插入图片描述在这里插入图片描述
在这里插入图片描述

新建工程

在这里插入图片描述

添加工具类

在添加工具类时可能会有错误提示,只是包名错了,修改报错文件的包成自己当前工程的包名即可解决问题:
在这里插入图片描述

添加设备参数列表

在AndroidManifest中声明指定的USB设备,设备信息存放在resource="@xml/device_filter"
在这里插入图片描述

<intent-filter><action android:name="android.hardware.usb.action.USB_DEVICE_ATTACHED"/>
</intent-filter>
<meta-dataandroid:name="android.hardware.usb.action.USB_DEVICE_ATTACHED"android:resource="@xml/device_filter"/>

xml文件夹下命名为device_filter.xml:
在这里插入图片描述

<?xml version="1.0" encoding="utf-8"?>
<resources><!-- 0x0403 / 0x6001: FTDI FT232R UART --><usb-device vendor-id="1027" product-id="24577" /><!-- 0x0403 / 0x6015: FTDI FT231X --><usb-device vendor-id="1027" product-id="24597" /><!-- 0x2341 / Arduino --><usb-device vendor-id="9025" /><!-- 0x16C0 / 0x0483: Teensyduino  --><usb-device vendor-id="5824" product-id="1155" /><!-- 0x10C4 / 0xEA60: CP210x UART Bridge --><usb-device vendor-id="4292" product-id="60000" /><!-- 0x067B / 0x2303: Prolific PL2303 --><usb-device vendor-id="1659" product-id="8963" />
</resources>

至于这个文件里面的数据表示的是什么,将USB转TTL模块插入电脑,在设备管理器里面可以看到:
在这里插入图片描述
PID_2303:2303的10进制是8963,也就是:product-id=“8963”
VID_067B:067B的10进制是1659,vendor-id=“1659”
不同的模块这两个值就不一样,所以就有了这样一个列表,当然这也不全,没有包含所有的型号。

具体实现

在MainActivity里面就可以写设备获取与数据收发了:

package com.example.otgdemo;import android.hardware.usb.UsbDevice;
import android.hardware.usb.UsbManager;
import android.os.AsyncTask;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.widget.Toast;import com.example.otgdemo.usbserial.driver.UsbSerialDriver;
import com.example.otgdemo.usbserial.driver.UsbSerialProber;
import com.example.otgdemo.usbserial.util.SerialInputOutputManager;
import com.example.otgdemo.utils.LogUtils;import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;public class MainActivity extends AppCompatActivity {private UsbManager mUsbManager;         // usb设备管理private List<DeviceEntry> mEntries;     // 串口设备列表private static UsbSerialDriver sDriver; // 打开的串口设备private static SerialInputOutputManager mSerialIoManager;   //数据发送、接收工具private ExecutorService mExecutorService;   //数据读取线程管理@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);// 获取USB_SERVICE的管理器mUsbManager = (UsbManager)getSystemService(this.USB_SERVICE);mEntries = new ArrayList<>();mExecutorService = Executors.newSingleThreadExecutor();refreshDeviceList();}/*** 刷新usb设备列表*/private void refreshDeviceList() {new AsyncTask<Void, Void, List<DeviceEntry>>() {@Overrideprotected List<DeviceEntry> doInBackground(Void... params) {Log.d("log","刷新设备列表 ...");final List<DeviceEntry> result = new ArrayList<>();Map<String, UsbDevice> deviceList = mUsbManager.getDeviceList();if (deviceList.isEmpty()) {Log.d("log","设备列表为空");} else {for (final UsbDevice device : mUsbManager.getDeviceList().values()) {final List<UsbSerialDriver> drivers = UsbSerialProber.probeSingleDevice(mUsbManager, device);Log.d("log","发现设备: " + device);if (drivers.isEmpty()) {Log.d("log","  - 空设备列表.");result.add(new DeviceEntry(null));} else {for (UsbSerialDriver driver : drivers) {Log.d("log","  + " + driver);result.add(new DeviceEntry(driver));}}}}return result;}@Overrideprotected void onPostExecute(List<DeviceEntry> result) {if (result.isEmpty()) {Toast.makeText(MainActivity.this, "没发现可用设备!", Toast.LENGTH_SHORT).show();return;}mEntries.clear();mEntries.addAll(result);sDriver = mEntries.get(0).driver;reStart();Log.d("log","停止刷新,发现" + mEntries.size() + " 个设备.");}}.execute((Void) null);}/*** 打开串口*/private void reStart() {if (sDriver == null) {Toast.makeText(this, "没有发现串口设备.", Toast.LENGTH_SHORT).show();} else {try {sDriver.open();sDriver.setParameters(57600, 8, UsbSerialDriver.STOPBITS_1, UsbSerialDriver.PARITY_NONE);} catch (IOException e) {LogUtils.d("设备打开错误: " + e.getMessage(), e);Toast.makeText(this, "设备打开错误: " + e.getMessage(), Toast.LENGTH_SHORT).show();try {sDriver.close();} catch (IOException e2) {// Ignore.}sDriver = null;return;}Toast.makeText(this, " 串口设备: " + sDriver.getClass().getSimpleName(), Toast.LENGTH_SHORT).show();}onDeviceStateChange();}/*** 重置串口*/private void onDeviceStateChange() {stopIoManager();startIoManager();}/*** 关闭串口*/private void stopIoManager() {if (mSerialIoManager != null) {LogUtils.d("Stopping io manager ..");mSerialIoManager.stop();mSerialIoManager = null;}}/*** 打开串口*/private void startIoManager() {if (sDriver != null) {LogUtils.d("Starting io manager ..");mSerialIoManager = new SerialInputOutputManager(sDriver, mListener);mExecutorService.submit(mSerialIoManager);//开启数据读取线程}}/*** Simple container for a UsbDevice and its driver.*/private static class DeviceEntry {public UsbSerialDriver driver;DeviceEntry(UsbSerialDriver driver) {this.driver = driver;}}/*** 数据读取回调*/private final SerialInputOutputManager.Listener mListener = new SerialInputOutputManager.Listener() {@Overridepublic void onRunError(Exception e) {LogUtils.d("Runner stopped.");}// 数据接收的回调函数@Overridepublic void onNewData(byte[] data, int len) {}};/*** 数据发送** @param data 要发送的数据*/public static void sendData(byte[] data) {if (mSerialIoManager != null) {mSerialIoManager.writeAsync(data);}}
}

WIFI adb

如果手机要调试那么会占用手机的usb端口,要下程序又要接OTG,来回拔插比较麻烦,可以在AS中使用wifi adb方式来部署应用。在file->seting->Plugins下选择下方中间一项:
在这里插入图片描述

右侧会有一个install的按钮,我之前安装过了就是这个样子:
在这里插入图片描述
安装完后会有这样一个图标:
在这里插入图片描述
在usb接到手机之后点击这个图标,出现如下,点击右侧的CONNECT即可:
在这里插入图片描述
变成这样就说明链接成功,就可以拔掉usb线了:
在这里插入图片描述
最新demo加入了CH340支持。

这篇关于Android AS下的OTG串口设备读写的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1093210

相关文章

Android Paging 分页加载库使用实践

《AndroidPaging分页加载库使用实践》AndroidPaging库是Jetpack组件的一部分,它提供了一套完整的解决方案来处理大型数据集的分页加载,本文将深入探讨Paging库... 目录前言一、Paging 库概述二、Paging 3 核心组件1. PagingSource2. Pager3.

Linux之platform平台设备驱动详解

《Linux之platform平台设备驱动详解》Linux设备驱动模型中,Platform总线作为虚拟总线统一管理无物理总线依赖的嵌入式设备,通过platform_driver和platform_de... 目录platform驱动注册platform设备注册设备树Platform驱动和设备的关系总结在 l

Android kotlin中 Channel 和 Flow 的区别和选择使用场景分析

《Androidkotlin中Channel和Flow的区别和选择使用场景分析》Kotlin协程中,Flow是冷数据流,按需触发,适合响应式数据处理;Channel是热数据流,持续发送,支持... 目录一、基本概念界定FlowChannel二、核心特性对比数据生产触发条件生产与消费的关系背压处理机制生命周期

Android ClassLoader加载机制详解

《AndroidClassLoader加载机制详解》Android的ClassLoader负责加载.dex文件,基于双亲委派模型,支持热修复和插件化,需注意类冲突、内存泄漏和兼容性问题,本文给大家介... 目录一、ClassLoader概述1.1 类加载的基本概念1.2 android与Java Class

C#读写文本文件的多种方式详解

《C#读写文本文件的多种方式详解》这篇文章主要为大家详细介绍了C#中各种常用的文件读写方式,包括文本文件,二进制文件、CSV文件、JSON文件等,有需要的小伙伴可以参考一下... 目录一、文本文件读写1. 使用 File 类的静态方法2. 使用 StreamReader 和 StreamWriter二、二进

MySQL主从复制与读写分离的用法解读

《MySQL主从复制与读写分离的用法解读》:本文主要介绍MySQL主从复制与读写分离的用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、主从复制mysql主从复制原理实验案例二、读写分离实验案例安装并配置mycat 软件设置mycat读写分离验证mycat读

Redis分片集群、数据读写规则问题小结

《Redis分片集群、数据读写规则问题小结》本文介绍了Redis分片集群的原理,通过数据分片和哈希槽机制解决单机内存限制与写瓶颈问题,实现分布式存储和高并发处理,但存在通信开销大、维护复杂及对事务支持... 目录一、分片集群解android决的问题二、分片集群图解 分片集群特征如何解决的上述问题?(与哨兵模

Android DataBinding 与 MVVM使用详解

《AndroidDataBinding与MVVM使用详解》本文介绍AndroidDataBinding库,其通过绑定UI组件与数据源实现自动更新,支持双向绑定和逻辑运算,减少模板代码,结合MV... 目录一、DataBinding 核心概念二、配置与基础使用1. 启用 DataBinding 2. 基础布局

Android ViewBinding使用流程

《AndroidViewBinding使用流程》AndroidViewBinding是Jetpack组件,替代findViewById,提供类型安全、空安全和编译时检查,代码简洁且性能优化,相比Da... 目录一、核心概念二、ViewBinding优点三、使用流程1. 启用 ViewBinding (模块级

Android学习总结之Java和kotlin区别超详细分析

《Android学习总结之Java和kotlin区别超详细分析》Java和Kotlin都是用于Android开发的编程语言,它们各自具有独特的特点和优势,:本文主要介绍Android学习总结之Ja... 目录一、空安全机制真题 1:Kotlin 如何解决 Java 的 NullPointerExceptio