android通過Genymotion模擬器枚舉U盤設(shè)備(一)

背景:

由于公司車機(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

image

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

image

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

image

完成配置后啟動模擬器,即可在模擬器完成與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)限");
                }
            }

        }
    }
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容