项目中有一个新的需求,要求可以连接一个USB体温枪,APP可以从体温枪中读取到体温数据,一番搜寻之后发现一个封装很棒的USB通信库。
github地址:usb-serial-for-android

准备工作

  1. 从 github 上 clone 这个库的源文件
  2. 在Android Studio中引入模块

  3. 在build.gradle中添加依赖compile project(path: ':usbSerialForAndroid')

  4. 复制 device_filter.xml 到项目的 res/xml/ 文件夹下

  5. 配置 AndroidManifest.xml 文件

<uses-feature android:name="android.hardware.usb.host" />
<application>
<activity
android:name="..."
...>
<intent-filter>
<action android:name="android.hardware.usb.action.USB_DEVICE_ATTACHED" />
</intent-filter>
<meta-data
android:name="android.hardware.usb.action.USB_DEVICE_ATTACHED"
android:resource="@xml/device_filter" />
</activity>
</application>

使用

示例代码解析:

// 获取系统服务得到UsbManager实例
UsbManager manager = (UsbManager) getSystemService(Context.USB_SERVICE);
//查找所有插入的设备
List<UsbSerialDriver> availableDrivers = UsbSerialProber.getDefaultProber().findAllDrivers(manager);
if (availableDrivers.isEmpty()) {
return;
} // 打开设备,建立通信连接
UsbSerialDriver driver = availableDrivers.get(0);
UsbDeviceConnection connection = manager.openDevice(driver.getDevice());
if (connection == null) {
// You probably need to call UsbManager.requestPermission(driver.getDevice(), ..)
return;
} //打开端口,设置端口参数,读取数据
UsbSerialPort port = driver.getPorts().get(0);
try {
port.open(connection);
//四个参数分别是:波特率,数据位,停止位,校验位
port.setParameters(115200, 8, UsbSerialPort.STOPBITS_1, UsbSerialPort.PARITY_NONE); byte buffer[] = new byte[16];
int numBytesRead = port.read(buffer, 1000);
Log.d(TAG, "Read " + numBytesRead + " bytes.");
} catch (IOException e) {
// Deal with error.
} finally {
port.close();
}

上述代码是库作者写的一段示例代码,可以看出使用上非常简单,下面的代码演示了如何将USB操作都封装到一个类里。

public class TemperatureUsbControl {

    private static final String TAG = TemperatureUsbControl.class.getSimpleName();
private static final String TEMPERATURE_USB_VENDOR_ID = "067B"; //供应商id
private static final String TEMPERATURE_USB_PRODUCT_ID = "2303"; //产品id
private Context mContext;
private UsbManager mUsbManager; //USB管理器
private UsbSerialPort sTemperatureUsbPort = null; //接体温枪的usb端口
private SerialInputOutputManager mSerialIoManager; //输入输出管理器(本质是一个Runnable)
private final ExecutorService mExecutor = Executors.newSingleThreadExecutor(); //用于不断从端口读取数据
//数据输入输出监听器
private final SerialInputOutputManager.Listener mListener =
new SerialInputOutputManager.Listener() { @Override
public void onRunError(Exception e) {
Log.d(TAG, "Runner stopped.");
} @Override
public void onNewData(final byte[] data) {
Log.d(TAG, "new data.");
}
}; public TemperatureUsbControl(Context context) {
mContext = context;
} public void initUsbControl() {
mUsbManager = (UsbManager) mContext.getSystemService(Context.USB_SERVICE);
//全部设备
List<UsbSerialDriver> usbSerialDrivers = UsbSerialProber.getDefaultProber().findAllDrivers(mUsbManager);
//全部端口
List<UsbSerialPort> usbSerialPorts = new ArrayList<UsbSerialPort>();
for (UsbSerialDriver driver : usbSerialDrivers) {
List<UsbSerialPort> ports = driver.getPorts();
Log.d(TAG, String.format("+ %s: %s port%s",
driver, Integer.valueOf(ports.size()), ports.size() == 1 ? "" : "s"));
usbSerialPorts.addAll(ports);
}
String vendorId;
String productId;
//校验设备,设备是 2303设备
for (UsbSerialPort port : usbSerialPorts) {
UsbSerialDriver driver = port.getDriver();
UsbDevice device = driver.getDevice();
vendorId = HexDump.toHexString((short) device.getVendorId());
productId = HexDump.toHexString((short) device.getProductId());
if (vendorId.equals(TEMPERATURE_USB_VENDOR_ID) && productId.equals(TEMPERATURE_USB_PRODUCT_ID)) {
sTemperatureUsbPort = port;
}
}
if (sTemperatureUsbPort != null) {
//成功获取端口,打开连接
UsbDeviceConnection connection = mUsbManager.openDevice(sTemperatureUsbPort.getDriver().getDevice());
if (connection == null) {
Log.e(TAG, "Opening device failed");
return;
}
try {
sTemperatureUsbPort.open(connection);
//设置波特率
sTemperatureUsbPort.setParameters(4800, 8, UsbSerialPort.STOPBITS_1, UsbSerialPort.PARITY_NONE); } catch (IOException e) {
//打开端口失败,关闭!
Log.e(TAG, "Error setting up device: " + e.getMessage(), e);
try {
sTemperatureUsbPort.close();
} catch (IOException e2) {
// Ignore.
}
sTemperatureUsbPort = null;
return;
}
} else {
//提示未检测到设备
}
} public void onDeviceStateChange() {
//重新开启USB管理器
stopIoManager();
startIoManager();
} private void startIoManager() {
if (sTemperatureUsbPort != null) {
Log.i(TAG, "Starting io manager ..");
mSerialIoManager = new SerialInputOutputManager(sTemperatureUsbPort, mListener);
mExecutor.submit(mSerialIoManager); //实质是用一个线程不断读取USB端口
}
} private void stopIoManager() {
if (mSerialIoManager != null) {
Log.i(TAG, "Stopping io manager ..");
mSerialIoManager.stop();
mSerialIoManager = null;
}
} public void onPause() {
stopIoManager();
if (sTemperatureUsbPort != null) {
try {
sTemperatureUsbPort.close();
} catch (IOException e) {
// Ignore.
}
sTemperatureUsbPort = null;
}
}
}

在Activity中使用:

    @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_punch);
initUsbControl(); //初始化USB控制器
} /**
* 初始化USB
*/
private void initUsbControl() {
mTemperatureUsbControl = new TemperatureUsbControl(mContext);
mTemperatureUsbControl.initUsbControl();
} @Override
protected void onResume() {
super.onResume();
IntentFilter usbFilter = new IntentFilter();
usbFilter.addAction(UsbManager.ACTION_USB_DEVICE_ATTACHED);
usbFilter.addAction(UsbManager.ACTION_USB_DEVICE_DETACHED);
registerReceiver(mUsbReceiver, usbFilter);
mTemperatureUsbControl.onDeviceStateChange();
} @Override
protected void onPause() {
super.onPause();
mTemperatureUsbControl.onPause();
unregisterReceiver(mUsbReceiver);
} /**
* 用于检测usb插入状态的BroadcasReceiver
*/
private final BroadcastReceiver mUsbReceiver = new BroadcastReceiver() {
public void onReceive(Context context, Intent intent) {
String action = intent.getAction();
if (UsbManager.ACTION_USB_DEVICE_ATTACHED.equals(action)) {
//设备插入
mTemperatureUsbControl.initUsbControl();
mTemperatureUsbControl.onDeviceStateChange();
Log.e(TAG, "ACTION_USB_DEVICE_ATTACHED");
} else if (UsbManager.ACTION_USB_DEVICE_DETACHED.equals(action)) {
//设备移除
mTemperatureUsbControl.onPause();
Log.e(TAG, "ACTION_USB_DEVICE_DETACHED");
}
}
};

读到的数据如果需要在Activity中使用,可以使用EventBus来完成~

一步一步教你简单完成 Android USB开发的更多相关文章

  1. 通过Dapr实现一个简单的基于.net的微服务电商系统(四)——一步一步教你如何撸Dapr之订阅发布

    之前的章节我们介绍了如何通过dapr发起一个服务调用,相信看过前几章的小伙伴已经对dapr有一个基本的了解了,今天我们来聊一聊dapr的另外一个功能--订阅发布 目录:一.通过Dapr实现一个简单的基 ...

  2. 通过Dapr实现一个简单的基于.net的微服务电商系统(三)——一步一步教你如何撸Dapr

    目录:一.通过Dapr实现一个简单的基于.net的微服务电商系统 二.通过Dapr实现一个简单的基于.net的微服务电商系统(二)--通讯框架讲解 三.通过Dapr实现一个简单的基于.net的微服务电 ...

  3. 通过Dapr实现一个简单的基于.net的微服务电商系统(五)——一步一步教你如何撸Dapr之状态管理

    状态管理和上一章的订阅发布都算是Dapr相较于其他服务网格框架来讲提供的比较特异性的内容,今天我们来讲讲状态管理. 目录:一.通过Dapr实现一个简单的基于.net的微服务电商系统 二.通过Dapr实 ...

  4. 通过Dapr实现一个简单的基于.net的微服务电商系统(六)——一步一步教你如何撸Dapr之Actor服务

    我个人认为Actor应该是Dapr里比较重头的部分也是Dapr一直在讲的所谓"stateful applications"真正具体的一个实现(个人认为),上一章讲到有状态服务可能很 ...

  5. 通过Dapr实现一个简单的基于.net的微服务电商系统(七)——一步一步教你如何撸Dapr之服务限流

    在一般的互联网应用中限流是一个比较常见的场景,也有很多常见的方式可以实现对应用的限流比如通过令牌桶通过滑动窗口等等方式都可以实现,也可以在整个请求流程中进行限流比如客户端限流就是在客户端通过随机数直接 ...

  6. 通过Dapr实现一个简单的基于.net的微服务电商系统(八)——一步一步教你如何撸Dapr之链路追踪

    Dapr提供了一些开箱即用的分布式链路追踪解决方案,今天我们来讲一讲如何通过dapr的configuration来实现非侵入式链路追踪的 目录:一.通过Dapr实现一个简单的基于.net的微服务电商系 ...

  7. 通过Dapr实现一个简单的基于.net的微服务电商系统(九)——一步一步教你如何撸Dapr之OAuth2授权

    Oauth2授权,熟悉微信开发的同学对这个东西应该不陌生吧.当我们的应用系统需要集成第三方授权时一般都会做oauth集成,今天就来看看在Dapr的语境下我们如何仅通过配置无需修改应用程序的方式让第三方 ...

  8. 通过Dapr实现一个简单的基于.net的微服务电商系统(十)——一步一步教你如何撸Dapr之绑定

    如果说Actor是dapr有状态服务的内部体现的话,那绑定应该是dapr对serverless这部分的体现了.我们可以通过绑定极大的扩展应用的能力,甚至未来会成为serverless的基础.最开始接触 ...

  9. 通过Dapr实现一个简单的基于.net的微服务电商系统(十一)——一步一步教你如何撸Dapr之自动扩/缩容

    上一篇我们讲到了dapr提供的bindings,通过绑定可以让我们的程序轻装上阵,在极端情况下几乎不需要集成任何sdk,仅需要通过httpclient+text.json即可完成对外部组件的调用,这样 ...

随机推荐

  1. 微信小程序(应用号)开发新闻客户端的实战课程

    摘要: 本实例将演示从零开发一个微信应用号的过程,页面轮播与跳转传值,实现单元格自定义布局,全部源码可通过github下载. 下载最新版的微信小程序开发工具,目前是v0.9.092300 下载地址:h ...

  2. MYSQL 的optimize怎么用

    当对表有大量的增删改操作时,需要用optimize对表进行优化.可以减少空间与提高I/O性能,命令optimize table tablename;假如有foo表且存储引擎为MyISAM. mysql ...

  3. MySQL数据库 常用命令

    1.MySQL常用命令 create database name;创建数据库 use databasename;选择数据库 drop database name 直接删除数据库,不提醒 show ta ...

  4. node.js ----NPM使用介绍

    NPM 使用介绍 NPM是随同NodeJS一起安装的包管理工具,能解决NodeJS代码部署上的很多问题,常见的使用场景有以下几种: 允许用户从NPM服务器下载别人编写的第三方包到本地使用. 允许用户从 ...

  5. jquery列表自动加载更多

    <!doctype html> <html> <head> <meta charset="utf-8"> <title> ...

  6. Netty(三):线程模型

    Netty中支持单线程模型,多线程模型,主从多线程模型. 1 单线程模型 在ServerBootstrap调用方法group的时候,传递的参数是同一个线程组,且在构造线程组的时候,构造参数为1,这种开 ...

  7. firework压缩图片类似于GD库中压缩图片的思路

    1.先建一张空白图片, 2.再把需要压缩的图片拖上去, 3.符合画布 4.调到需要的大小

  8. jquery元素分组插件,用于把一连串元素分成多组,如把多个a标签分成多组放入<li>元素中,可以用于简化多图滚动为一个元素滚动,兼容ie6

    三个参数 <script type="text/javascript"> /* *sclass:设置包裹元素的类 * packages:设置包裹的元素 * row:设置 ...

  9. Linux U盘只读解决方法

    Linux Fat的U盘只读,这个问题经常出现,原因大家都说了是U盘的错误,出现这种情况后,一般的解决方案是 mount | grep <U盘的标签> # 找到你的U盘的对应的设备名称,如 ...

  10. FreeRTOS在神舟IV号开发板的应用demo

    下面一个可以直接编译运行的例子,FreeRTOS的版本是V7.1.0,芯片是STM32F107VCT6,使用的开发环境是Keil uVision5. 这里例子创建了四个任务,每个任务控制一个LED的亮 ...