從android2.3開始支持nfc。不過nfc應(yīng)用只能在android手機(或平板電腦)上測試和開發(fā),而且android手機還必須有nfc芯 片。而且如果測試nfc傳輸文件時至少需要兩部支持nfc的手機。當(dāng)然,如果測試讀寫nfc標(biāo)簽,還需要一些nfc標(biāo)簽或帖子。而且nfc在模擬器上時不能運行的。所以從這一點來說,nfc開發(fā)需要更多的設(shè)備,比較麻煩。這也藍(lán)牙、傳感器是一樣的。都不能在android模擬器上開發(fā)和測試。真不知道google為什么不解決這一問題。
不過這種問題也不是不能解決,而且并不復(fù)雜。既然模擬器沒有提供這樣的功能。我們可以將nfc功能模擬出來(實際上,藍(lán)牙、傳感器都可以進(jìn)行模擬,可能很 多讀者用過一些傳感器模擬軟件,nfc模擬和這個類似)。而且要求是與真實的nfc環(huán)境無縫對接。也就是說,使用模擬nfc功能開發(fā)的android應(yīng)用 可以不需要修改一行代碼,甚至不需要重新編譯,就可以直接用在真實的nfc環(huán)境。
那么怎么解決這個問題呢?要想知道如何模擬nfc,需要先從宏觀上了解nfc的工作原理。這里用nfc標(biāo)簽作為例子。nfc數(shù)據(jù)傳輸和這個類似。當(dāng)nfc 標(biāo)簽靠近手機時,android系統(tǒng)中有一個叫nfc的系統(tǒng)應(yīng)用(在<android源代碼根目錄>/pakcages/apps/nfc目 錄中),該應(yīng)用會發(fā)送一個activity action,該action會調(diào)用一個在系統(tǒng)中注冊的用于處理nfc請求的窗口(如果沒有,就調(diào)用nfc應(yīng)用中默認(rèn)的處理窗口)。這里的關(guān)鍵是 activity action。既然nfc應(yīng)用會發(fā)出一個activity action,那么用于模擬nfc的程序,也發(fā)送一個activity action,不就可以共享用于處理nfc請求的窗口(由用戶建立的activity)了嗎?只要在發(fā)送activity action時加一個標(biāo)志,就可以區(qū)分是activity action是nfc系統(tǒng)應(yīng)用發(fā)出的,還是模擬nfc的程序發(fā)出的。這樣在處理請求的nfc窗口中就可以根據(jù)不同的情況進(jìn)行處理。為了更透明,可以編寫一 個activity類(如nfcactivity),該類根據(jù)這兩種情況進(jìn)行處理。又因為不管是哪種情況,都需要提供寫入nfc標(biāo)簽的數(shù)據(jù),或接收nfc 標(biāo)簽中的數(shù)據(jù)。所以可以在該nfcactivity中提供一些回調(diào)方法,當(dāng)需要提供或接收數(shù)據(jù)時,調(diào)用這些方法即可。最后需要使用nfc功能時,用戶自己 編寫的窗口類只需要從nfcactivity類繼承,并實現(xiàn)相應(yīng)的回調(diào)方法即可。例如,下面就是一個實現(xiàn)方法,可接收nfc標(biāo)簽的數(shù)據(jù),并寫入新數(shù)據(jù)。
public class nfcmantestactivity extends nfcactivity
{
private nfcman mnfcman;
private edittext nfctagtext;
@override
protected void oncreate(bundle savedinstancestate)
{
super.oncreate(savedinstancestate);
setcontentview(r.layout.activity_nfcman_test);
nfctagtext = (edittext) findviewbyid(r.id.edittext_nfc);
mnfcman = new nfcman(this);
}
// 當(dāng)nfc標(biāo)簽靠近手機時調(diào)用,data用于接收nfc標(biāo)簽中的數(shù)據(jù),
// 返回值會重新寫到nfc標(biāo)簽中
@override
public string onnear(nfcdata data)
{
// 返回要寫入nfc標(biāo)簽的文本
return nfctagtext.gettext().tostring();
}
// 將數(shù)據(jù)成功寫入nfc標(biāo)簽后會調(diào)用該方法
@override
public void onnfcsuccess()
{
// 顯示成功寫入數(shù)據(jù)的提示
toast.maketext(this, "成功寫入數(shù)據(jù)", toast.length_long).show();
}
// 寫入數(shù)據(jù)失敗后調(diào)用該方法
@override
public void onnfcfailed(int error, string errormsg)
{
// 顯示寫入數(shù)據(jù)異常的編碼和信息
toast.maketext(this, "error:" + error + "\n" + "msg:" + errormsg,
toast.length_long).show();
}
}
從nfcmantestactivity類的代碼可以看出,共有如下三個回調(diào)方法。分別處理讀寫nfc標(biāo)簽數(shù)據(jù),寫入成功和寫入失敗三個事件。
onnear
onnfcsuccess
onnfcfailed
從這一點可以看出,完全隱藏了nfc的影子。下面用圖1描述一下這個nfc模擬系統(tǒng)的原理。
圖1 nfc模擬器的實現(xiàn)原理
從圖1可以看出,需要有一套虛擬的nfc標(biāo)簽,這些是在pc上用軟件模擬的。對于虛擬nfc標(biāo)簽,通過socket與安裝在nfc設(shè)備中的模擬nfc處理 程序進(jìn)行交互,然后該程序會發(fā)送broadcast action,最后接收到這個broadcast后,會繼續(xù)調(diào)用activity action調(diào)用處理nfc請求的窗口。這一點與nfc系統(tǒng)程序一樣,只不過中間多了一個發(fā)送broadcast的過程。因為模擬nfc處理程序與處理 nfc請求的nfcactivity所在的library是分離的,只能通過broadcast進(jìn)行通知。而nfc系統(tǒng)程序是通過nfc驅(qū)動感知真實 nfc標(biāo)簽是否靠近的。而對于處理nfc請求的窗口來說,不管是虛擬的nfc標(biāo)簽,還是真實的nfc標(biāo)簽,都處理同一個activity action。所以處理nfc請求的activity可以共用一套代碼。
我將這個模擬nfc的應(yīng)用稱為nfcman(nfc俠),下面看一下該應(yīng)用如何模擬nfc標(biāo)簽,模擬界面如圖2所示。
圖2 nfc模擬器主界面
在圖2中上面是模擬的nfc設(shè)備。只要android手機(不需要支持nfc)或android模擬器上運行的模擬nfc的應(yīng)用(如圖3所示)根據(jù)圖2中右下角的ip連接到nfc模擬器,就會顯示一個手機圖標(biāo)。
圖3
圖2的下面是虛擬的nfc標(biāo)簽,點擊右下角的“新建nfc標(biāo)簽”可以建立新的nfc標(biāo)簽。雙擊nfc標(biāo)簽會顯示該標(biāo)簽的信息,如圖4所示。這些信息包括標(biāo)簽名稱、最大容量、標(biāo)簽內(nèi)容(默認(rèn)是空)。只要將虛擬nfc標(biāo)簽拖動到上面的虛擬nfc設(shè)備,就相當(dāng)于將nfc標(biāo)簽靠近了該設(shè)備。然后會通過socket將相應(yīng)的數(shù)據(jù)傳到圖3所示的android應(yīng)用。接著該應(yīng)用會發(fā)送broadcast。最后接收到該broadcast的android應(yīng)用會通過activity action調(diào)用處理nfc請求的activity。會根據(jù)情況調(diào)用onnear、onnfcsuccess、onnfcfailed三個回調(diào)方法。
圖4