一步一步教你简单完成 Android USB开发
项目中有一个新的需求,要求可以连接一个USB体温枪,APP可以从体温枪中读取到体温数据,一番搜寻之后发现一个封装很棒的USB通信库。
github地址:usb-serial-for-android
准备工作
- 从 github 上 clone 这个库的源文件
在Android Studio中引入模块
在build.gradle中添加依赖
compile project(path: ':usbSerialForAndroid')
复制 device_filter.xml 到项目的 res/xml/ 文件夹下
配置 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开发的更多相关文章
- 通过Dapr实现一个简单的基于.net的微服务电商系统(四)——一步一步教你如何撸Dapr之订阅发布
之前的章节我们介绍了如何通过dapr发起一个服务调用,相信看过前几章的小伙伴已经对dapr有一个基本的了解了,今天我们来聊一聊dapr的另外一个功能--订阅发布 目录:一.通过Dapr实现一个简单的基 ...
- 通过Dapr实现一个简单的基于.net的微服务电商系统(三)——一步一步教你如何撸Dapr
目录:一.通过Dapr实现一个简单的基于.net的微服务电商系统 二.通过Dapr实现一个简单的基于.net的微服务电商系统(二)--通讯框架讲解 三.通过Dapr实现一个简单的基于.net的微服务电 ...
- 通过Dapr实现一个简单的基于.net的微服务电商系统(五)——一步一步教你如何撸Dapr之状态管理
状态管理和上一章的订阅发布都算是Dapr相较于其他服务网格框架来讲提供的比较特异性的内容,今天我们来讲讲状态管理. 目录:一.通过Dapr实现一个简单的基于.net的微服务电商系统 二.通过Dapr实 ...
- 通过Dapr实现一个简单的基于.net的微服务电商系统(六)——一步一步教你如何撸Dapr之Actor服务
我个人认为Actor应该是Dapr里比较重头的部分也是Dapr一直在讲的所谓"stateful applications"真正具体的一个实现(个人认为),上一章讲到有状态服务可能很 ...
- 通过Dapr实现一个简单的基于.net的微服务电商系统(七)——一步一步教你如何撸Dapr之服务限流
在一般的互联网应用中限流是一个比较常见的场景,也有很多常见的方式可以实现对应用的限流比如通过令牌桶通过滑动窗口等等方式都可以实现,也可以在整个请求流程中进行限流比如客户端限流就是在客户端通过随机数直接 ...
- 通过Dapr实现一个简单的基于.net的微服务电商系统(八)——一步一步教你如何撸Dapr之链路追踪
Dapr提供了一些开箱即用的分布式链路追踪解决方案,今天我们来讲一讲如何通过dapr的configuration来实现非侵入式链路追踪的 目录:一.通过Dapr实现一个简单的基于.net的微服务电商系 ...
- 通过Dapr实现一个简单的基于.net的微服务电商系统(九)——一步一步教你如何撸Dapr之OAuth2授权
Oauth2授权,熟悉微信开发的同学对这个东西应该不陌生吧.当我们的应用系统需要集成第三方授权时一般都会做oauth集成,今天就来看看在Dapr的语境下我们如何仅通过配置无需修改应用程序的方式让第三方 ...
- 通过Dapr实现一个简单的基于.net的微服务电商系统(十)——一步一步教你如何撸Dapr之绑定
如果说Actor是dapr有状态服务的内部体现的话,那绑定应该是dapr对serverless这部分的体现了.我们可以通过绑定极大的扩展应用的能力,甚至未来会成为serverless的基础.最开始接触 ...
- 通过Dapr实现一个简单的基于.net的微服务电商系统(十一)——一步一步教你如何撸Dapr之自动扩/缩容
上一篇我们讲到了dapr提供的bindings,通过绑定可以让我们的程序轻装上阵,在极端情况下几乎不需要集成任何sdk,仅需要通过httpclient+text.json即可完成对外部组件的调用,这样 ...
随机推荐
- linux过滤ip段
https://www.2cto.com/net/201307/227257.html
- UDP通信接收端,接收二维数组,内容为0与1
1: using System; 2: using System.Net; 3: using System.Net.Sockets; 4: using System.Text; 5: 6: 7 ...
- mongodb: 安装 建/删 库,表
mongodb的安装 下载mongodb www.mongodb.org 下载最新stable版 解压文件 3.不用编译,解压之后本身就是编译后的二进制可执行文件 解压之后,目录格式如下 在bin目录 ...
- IIS配置asp.net网站
http://wenku.baidu.com/view/f8ce6c14767f5acfa0c7cd36.html
- Bootstrp--一个导航面板切换的实用例子
<!--导航区开始--> <ul class="nav nav-tabs nav-stacked" role="tablist"> &l ...
- 摩托罗拉SE955 One Discrete Length,Two Discrete Lengths,Length Within Range 相关解释
motorola scanner datasheet相关解释(下面通过Simple Serial Interface(SSI)进行设置,非扫描官方datasheet的设置条码): One Discre ...
- 为什么引入TSS
[0]README text description from orange's implemention of a os and for complete code ,please visit ht ...
- 基于imgAreaSelect的用户图像截取
前言:想到用户资料中一般有个图像自我截取的部分,为什么要截取呢,因为好看了.so,经过我各种百度,各种参考,终于打工搞成了,写下纪念纪念,让以后拿来就用也好. 一:想前端ui这东西,我就懒得说话了,哎 ...
- EasyPlayerPro Windows播放器全屏模式下GDI显示出现黑屏问题解决
问题来源 2017.12.21 前天有杭州某教育领域客户反馈有部分视频源在全屏模式下显示黑屏: 问题复现 EasyPlayerPro由于没有实现单个窗口完全全屏,故没有暴露该问题,晚上加班,加上单个窗 ...
- EasyPlayer播放海康大华RTSP流时RTSPClient客户端连接兼容问题的解决
在之前的博客<EasyPlayer RTSP播放器对RTSP播放地址url的通用兼容修改意见>中,我描述了遇到的一个客户在播放大华某款摄像机时地址不兼容的问题,这不,团队刚刚参考我的这个意 ...