---恢復內容開始---
想要開發一個通過藍牙傳輸數據的安卓應用程序,理所當然你需要從安卓開發者的藍牙頁面開始,那里面詳盡描述了所有必需的步驟,包括設備搜尋、配對、客戶端/服務器套接字和RFCOMM通道等。
嘿,嘿,別急,在開始討論套接字和線程編程之前還需要先執行一個基礎的藍牙操作,讓我們來考慮一個取巧的做法,基於安卓的一個最重要的特性:設備的默認藍牙應用提供的由一個給定應用發送用戶到另一個應用。這樣做將由安卓系統會替我們完成所有底層工作。
import android.bluetooth.BluetoothAdapter; //... // 函數中代碼 // 檢查設備是否支持藍牙
<!--EndFragment-->BluetoothAdapter btAdapter = BluetoothAdapter.getDefaultAdapter(); if (btAdapter == null) { // 設備不支持藍牙 // 告訴用戶我們無能為力 }
以上是我們需要執行的第一次檢查。完成之后,讓我們看看它如何用我們的應用啟動藍牙。
在過去的一篇關於手機短信編程的文章里,我們談到關於隱式意圖(Implicit Intent )允許我們指定希望系統處理的動作。 然后安卓會顯示一個單選列表,列出所有能夠完成我們要求的應用。下面是一段示例代碼:
// 啟動Android選擇器 Intent intent = new Intent(); intent.setAction(Intent.ACTION_SEND); intent.setType("text/plain"); intent.putExtra(Intent.EXTRA_STREAM, Uri.fromFile(file_to_transfer) ); //... startActivity(intent);
在這段代碼中,我們要求安卓系統發送一個文本文件。然后系統會顯示所有已安裝能夠處理這一意圖的活動列表,如下圖所示:
我們可以看到,這個基礎藍牙應用在這個列表的最上端。我們可以點擊它,當然也可以讓用戶選擇其他活動去完成傳送。我想,如果我們增加一點用戶友好體驗,可以自動啟動那個藍牙應用,而不是簡單地丟出一張列表甩在用戶臉上,把它和其他不必要的應用魚龍混雜在一起讓用戶玩找茬……要怎么做呢?
一個辦法是使用安卓的PackageManager,如下面代碼所示:
// 打開對應的列表 PackageManager pm = getPackageManager(); List appsList = pm.queryIntentActivities( intent, 0); if(appsList.size() > 0 { // 處理 }
PackageManger中的queryIntentActivities()方法返回一個列表,其中內容正是我們前面看到的那張圖中的所有選項,用迭代器遍歷這個列表每個ResolveInfo對象里封裝着我們需要的信息:
// 選擇藍牙 String packageName = null; String className = null; boolean found = false; for(ResolveInfo info: appsList){ packageName = info.activityInfo.packageName; if( packageName.equals("com.android.bluetooth")){ className = info.activityInfo.name; found = true; break;// 找到,結束查找 } } if(! found){ Toast.makeText(this, R.string.blu_notfound_inlist, Toast.LENGTH_SHORT).show(); // 退出 }
這段代碼之后,我們擁有了自動調用藍牙活動的必要信息:
// 設置啟動藍牙intent intent.setClassName(packageName, className); startActivity(intent);
我們所做的是使用包和其對應的類(在前面檢索到的)。我們會對上面的代碼很好奇,可能不知道“com.android.bluetooth”包的類名。如果我們將其打印出來會看到這些:“com.broadcom.bt.app.opp.OppLauncherActivity“OPP代表對象推送模式,是允許無線共享文件的系統組件。
看起來功德圓滿了,但為了上述所有代碼真正能夠使用,藍牙不僅僅需要設備支持也要由用戶在設置里啟用。那么我們想要做的第一件事情之一,就是要求用戶啟用藍牙一段時間(有多長?我們認為必要的那么長,在這里是300秒):
import android.bluetooth.BluetoothAdapter; //... // 設備可被發現持續時間
<!--EndFragment-->private static final int DISCOVER_DURATION = 300; // 請求碼<span style="font-family: Calibri;">(request code),</span><span style="font-family: 宋體;">必須大於0</span>
<!--EndFragment-->private static final int REQUEST_BLU = 1; //... public void enableBlu(){ // 啟動設備搜索——<span style="font-family: Calibri;"> </span><span style="font-family: 宋體;">這會自動啟用藍牙</span>
<!--EndFragment-->Intent discoveryIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_DISCOVERABLE); discoveryIntent.putExtra(BluetoothAdapter.EXTRA_DISCOVERABLE_DURATION, DISCOVER_DURATION ); startActivityForResult(discoveryIntent, REQUEST_BLU); }
一旦我們指定我們希望從我們的活動得到一個返回值,調用startActivityForResult()用戶會看到下面的對話框:
現在每當活動結束后會返回請求碼,即我們已送出的第二參數:REQUEST_BLU,同時返回數據和我們的主活動通過onActivityResult()方法返回的結果碼。我們知道請求碼,因此必須核對。至於結果代碼?很簡單:如果用戶回答“否”,上述要求的權限(或如果出現一個錯誤),結果碼將會是RESULT_CANCELED。相反地如果用戶接受,藍牙文檔指出結果碼等於這設備處於可被發現狀態的(即DISCOVER_DURATION,即300)的持續時間。
處理上述藍牙對話框的代碼如下:
// startActivityForResult執行完成... protected void onActivityResult (int requestCode, int resultCode, Intent data) { if (resultCode == DISCOVER_DURATION && requestCode == REQUEST_BLU) { // 這里編寫你的代碼 } else{ // 取消或出錯 Toast.makeText(this, R.string.blu_cancelled, Toast.LENGTH_SHORT).show(); } }
以下是流程圖,基本上解釋了我們在做什么:
大功告成了嗎?差不多是。最后,我們需要在Mainfest文件里請求藍牙權限:
<uses-permission android:name="android.permission.BLUETOOTH" /> <uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />
好了,一切准備妥當。為了測試這一切,我們需要使用至少兩款安卓設備,一個用於文件發送(安裝我們的應用程序),其他接收設備支持BT。無圖無真相,下面是幾張屏幕截圖。首先是負責發送的那台手機:
然后是對應的接收手機:
需要注意的是一旦接收這邊接受連接,接收的文件(kmemo.dat)會保存在SD卡上的BlueTooth文件夾里。所有底層數據傳輸都被安卓系統接管了。
版權聲明:原文鏈接: javacodegeeks 首發於: ImportNew.com - by 邢 敏。歡迎關注我的weibo:http://weibo.com/laowangba