背景:
由于公司車機(jī)插上U盤后讀寫后,U盤經(jīng)常會損壞,因此對這塊有點(diǎn)興趣,就研究了下U盤這塊內(nèi)容。
通過Genymotion 與 VirtualBox 可以實(shí)現(xiàn)將電腦中的USB設(shè)備轉(zhuǎn)接到Android模擬器中進(jìn)行通信。
1.1 Genymotion 配置
首先從https://www.genymotion.com/download/ 下載 Genymotion with VirtualBox

1.2 在完成模擬器的創(chuàng)建后,無論是否啟動模擬器都可以在 Oracle VM VirtualBox 中進(jìn)行USB的配置,如下圖:

1.3 選擇USB設(shè)備,插入U(xiǎn)盤,選擇添加設(shè)備,選擇你插入的U盤,這里我插入的u盤是SanDisk Ultra USB 3.0

完成配置后啟動模擬器,即可在模擬器完成與USB設(shè)備的通信。上述過程可以在模擬器已啟動時也可以操作,操作完成后需要重新拔插USB設(shè)備。
Android 枚舉設(shè)備
2.1、枚舉設(shè)備
在Android SDK中已經(jīng)完成了枚舉的封裝。Android提供兩種方式進(jìn)行枚舉:
2.1.1、通過Intent過濾器
當(dāng)USB設(shè)備連接時,Android會啟動包含:android.hardware.usb.action.USB_DEVICE_ATTACHED Action的意圖,我們可以在Manifest中注冊某個Activity支持處理該Action,讓系統(tǒng)自動將連接設(shè)備的抽象對象UsbDevice 發(fā)送至我們注冊的Activity:
<activity
android:name=".UsbActivity"
android:exported="true">
<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">
</meta-data>
</activity>
除了指定 Intent 過濾器 之外,還需要指定一個資源文件來指定支持處理的USB設(shè)備的屬性,在工程res/xml目錄下創(chuàng)建 device_filter.xml 文件,文件內(nèi)容如下:
<?xml version="1.0" encoding="utf-8"?>
<resources xmlns:android="http://schemas.android.com/apk/res/android">
<!--聲明可以處理所有usb設(shè)備-->
<usb-device />
<!--以下聲明處理指定處理vid和pid的usb設(shè)備 vendor-id 供應(yīng)商id 唯一 ,由供應(yīng)商向USB-if申請;
pid:產(chǎn)品id,由廠商自行決定 -->
<!--
<usb-devices vendor-id="11111" product-id="2222"/>
-->
</resources>
配置完成后,在USB設(shè)備加入時,就能通過以下方式從 Intent 獲取代表所連接設(shè)備的 UsbDevice
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
UsbDevice device = getIntent().getParcelableExtra(UsbManager.EXTRA_DEVICE);
}
@Override
protected void onNewIntent(Intent intent) {
super.onNewIntent(intent);
UsbDevice device = intent.getParcelableExtra(UsbManager.EXTRA_DEVICE);
}
2.1.2、主動枚舉設(shè)備
除了使用Intent過濾器在設(shè)備插入時接收連接設(shè)備的 UsbDevice之外,還可以使用 getDeviceList() 方法獲取連接的所有 USB 設(shè)備的哈希映射。
UsbManager usbManager = (UsbManager) getSystemService(Context.USB_SERVICE);
...
HashMap<String,UsbDevice> hashMap = usbManager.getDeviceList();
Collection<UsbDevice> values = hashMap.values();
Iterator<UsbDevice> iterator = values.iterator();
Log.i(Tag," iterator "+ iterator);
while (iterator.hasNext()){
UsbDevice usbDevice = iterator.next();
Log.i(Tag," device "+ usbDevice);
usbDevices.add(usbDevice);
}
2.2、獲得通信權(quán)限
如果應(yīng)用使用 Intent 過濾器來發(fā)現(xiàn)已連接的 USB 設(shè)備,則它會在用戶允許應(yīng)用處理 Intent 時自動獲得權(quán)限。否則,必須在應(yīng)用中明確請求權(quán)限,然后才能連接到設(shè)備。因此在嘗試與設(shè)備通信之前,必須先檢查是否具有訪問設(shè)備的權(quán)限。
如果還未具備設(shè)備訪問權(quán)限則需要通過 requestPermission() 完成請求:
public class UsbActivity extends AppCompatActivity implements UsbListAdapter.OnItemClickListener {
private final String TAG = UsbActivity.class.getSimpleName();
private List<UsbDevice> devices;
private RecyclerView usbRecyclerView;
private UsbListAdapter usbListAdapter;
private UsbManager usbManager;
private UsbReceiver usbReceiver;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_usb);
registerReceiver();
}
@Override
protected void onDestroy() {
super.onDestroy();
unRegisterReceiver();
}
@Override
public void onItemClick(View view, int position) {
UsbDevice usbDevice = devices.get(position);
Log.i(TAG, "onItemClick " + position);
if (!usbManager.hasPermission(usbDevice)) {//是否有權(quán)限
//申請權(quán)限,系統(tǒng)彈出對話框,用戶選擇后發(fā)出廣播,觸發(fā)注冊的廣播接收者
PendingIntent pendingIntent = PendingIntent.getBroadcast(getBaseContext(), 0,
new Intent(UsbReceiver.ACTION_USB_PERMISSION), 0);
usbManager.requestPermission(usbDevice, pendingIntent);
} else {
intentDetailActivity(usbDevice);
}
}
private void registerReceiver() {
usbReceiver = new UsbReceiver();
IntentFilter intentFilter = new IntentFilter();
intentFilter.addAction(UsbReceiver.ACTION_USB_PERMISSION);
registerReceiver(usbReceiver, intentFilter);
}
private void unRegisterReceiver() {
unregisterReceiver(usbReceiver);
}
private void intentDetailActivity(UsbDevice usbDevice){
Intent intent = new Intent(this, UsbDetailActivity.class);
intent.putExtra("usbDevice", usbDevice);
startActivity(intent);
}
private class UsbReceiver extends BroadcastReceiver {
public static final String ACTION_USB_PERMISSION = "com.fch.car.USB_PERMISSION";
@Override
public void onReceive(Context context, Intent intent) {
String action = intent.getAction();
if (TextUtils.equals(action, ACTION_USB_PERMISSION)) {
UsbDevice usbDevice = intent.getParcelableExtra(UsbManager.EXTRA_DEVICE);
if (intent.getBooleanExtra(UsbManager.EXTRA_PERMISSION_GRANTED, false)) {
Log.i(TAG, "獲得usb權(quán)限");
intentDetailActivity(usbDevice);
} else {
Log.i(TAG, "拒絕usb權(quán)限");
}
}
}
}
}