Android USB 通信开发

Android USB 通信开发主要涉及两种模式:主机模式(Host Mode)和配件模式(Accessory Mode)。以下是开发USB通信应用的关键知识点和步骤。

1. 基本概念

主机模式(Host Mode)

  • Android设备作为USB主机,控制连接的USB设备

  • 需要设备支持USB主机功能(通常需要OTG支持)

配件模式(Accessory Mode)

  • Android设备作为USB配件被其他主机控制

  • 需要设备支持USB配件模式

2. 开发准备

清单文件配置

XML 复制代码
<!-- 声明USB主机功能 -->
<uses-feature android:name="android.hardware.usb.host" />

<!-- 声明USB配件功能(如果需要) -->
<uses-feature android:name="android.hardware.usb.accessory" />

<!-- 添加必要的权限 -->
<uses-permission android:name="android.permission.USB_PERMISSION" />

3. USB主机模式开发

发现连接的USB设备

java 复制代码
UsbManager usbManager = (UsbManager) getSystemService(Context.USB_SERVICE);
HashMap<String, UsbDevice> deviceList = usbManager.getDeviceList();

请求设备权限

java 复制代码
private static final String ACTION_USB_PERMISSION = "com.example.USB_PERMISSION";

PendingIntent permissionIntent = PendingIntent.getBroadcast(this, 0, 
    new Intent(ACTION_USB_PERMISSION), PendingIntent.FLAG_IMMUTABLE);

IntentFilter filter = new IntentFilter(ACTION_USB_PERMISSION);
registerReceiver(usbReceiver, filter);

usbManager.requestPermission(device, permissionIntent);

广播接收器处理权限结果

java 复制代码
private final BroadcastReceiver usbReceiver = new BroadcastReceiver() {
    public void onReceive(Context context, Intent intent) {
        String action = intent.getAction();
        if (ACTION_USB_PERMISSION.equals(action)) {
            synchronized (this) {
                UsbDevice device = intent.getParcelableExtra(UsbManager.EXTRA_DEVICE);
                if (intent.getBooleanExtra(UsbManager.EXTRA_PERMISSION_GRANTED, false)) {
                    if(device != null){
                        // 权限已授予,可以通信
                        openDevice(device);
                    }
                } else {
                    Log.d(TAG, "权限被拒绝 for device " + device);
                }
            }
        }
    }
};

打开设备并通信

java 复制代码
private void openDevice(UsbDevice device) {
    UsbInterface intf = device.getInterface(0);
    UsbEndpoint endpoint = intf.getEndpoint(0);
    
    UsbDeviceConnection connection = usbManager.openDevice(device);
    if (connection == null) return;
    
    connection.claimInterface(intf, true);
    
    // 发送数据
    byte[] sendData = "Hello".getBytes();
    int sent = connection.bulkTransfer(endpoint, sendData, sendData.length, 0);
    
    // 接收数据
    byte[] receiveData = new byte[64];
    int received = connection.bulkTransfer(endpoint, receiveData, receiveData.length, 0);
}

4. USB配件模式开发

清单文件配置

java 复制代码
<activity ...>
    <intent-filter>
        <action android:name="android.hardware.usb.action.USB_ACCESSORY_ATTACHED" />
    </intent-filter>
    <meta-data 
        android:name="android.hardware.usb.action.USB_ACCESSORY_ATTACHED"
        android:resource="@xml/accessory_filter" />
</activity>

配件过滤器(xml/accessory_filter.xml)

XML 复制代码
<resources>
    <usb-accessory 
        manufacturer="YourCompany" 
        model="YourModel" 
        version="1.0" />
</resources>

检测配件连接

java 复制代码
UsbManager usbManager = (UsbManager) getSystemService(Context.USB_SERVICE);
UsbAccessory[] accessoryList = usbManager.getAccessoryList();

if (accessoryList != null && accessoryList.length > 0) {
    UsbAccessory accessory = accessoryList[0];
    openAccessory(accessory);
}

打开配件通信

java 复制代码
private void openAccessory(UsbAccessory accessory) {
    ParcelFileDescriptor fd = usbManager.openAccessory(accessory);
    if (fd != null) {
        FileDescriptor fileDescriptor = fd.getFileDescriptor();
        FileInputStream inputStream = new FileInputStream(fileDescriptor);
        FileOutputStream outputStream = new FileOutputStream(fileDescriptor);
        
        // 现在可以通过输入输出流进行通信
    }
}

5. 常见问题

  1. 权限问题:确保正确请求和处理USB权限

  2. 设备兼容性:并非所有Android设备都支持USB主机模式

  3. 线程阻塞:USB通信可能阻塞UI线程,建议在后台线程进行

  4. 连接稳定性:处理USB设备插拔事件

6. 高级主题

  • USB串口通信:使用CDC(Communication Device Class)协议

  • HID设备通信:与键盘、鼠标等HID设备交互

  • 同步通信:使用控制传输(Control Transfer)和中断传输(Interrupt Transfer)

7. 推荐库

开发时请参考Android官方USB文档获取最新信息。

相关推荐
寻星探路14 小时前
【深度长文】万字攻克网络原理:从 HTTP 报文解构到 HTTPS 终极加密逻辑
java·开发语言·网络·python·http·ai·https
工程师老罗16 小时前
如何在Android工程中配置NDK版本
android
曹牧17 小时前
Spring Boot:如何测试Java Controller中的POST请求?
java·开发语言
爬山算法17 小时前
Hibernate(90)如何在故障注入测试中使用Hibernate?
java·后端·hibernate
kfyty72517 小时前
集成 spring-ai 2.x 实践中遇到的一些问题及解决方案
java·人工智能·spring-ai
猫头虎18 小时前
如何排查并解决项目启动时报错Error encountered while processing: java.io.IOException: closed 的问题
java·开发语言·jvm·spring boot·python·开源·maven
李少兄18 小时前
在 IntelliJ IDEA 中修改 Git 远程仓库地址
java·git·intellij-idea
忆~遂愿18 小时前
ops-cv 算子库深度解析:面向视觉任务的硬件优化与数据布局(NCHW/NHWC)策略
java·大数据·linux·人工智能
小韩学长yyds18 小时前
Java序列化避坑指南:明确这4种场景,再也不盲目实现Serializable
java·序列化
仟濹18 小时前
【Java基础】多态 | 打卡day2
java·开发语言