一 你需要準(zhǔn)備的:一部有nfc的手機(jī),一張有nfc標(biāo)簽的卡
二 nfc簡(jiǎn)介
nfc(近距離無(wú)線通訊技術(shù)),是由非接觸式射頻識(shí)別(RFID)及互連互通技術(shù)整合演變而來(lái),通過(guò)在單一芯片上集成感應(yīng)式讀卡器、感應(yīng)式卡片和點(diǎn)對(duì)點(diǎn)通信的功能,利用移動(dòng)終端實(shí)現(xiàn)移動(dòng)支付、電子票務(wù)、門(mén)禁、移動(dòng)身份識(shí)別、防偽等應(yīng)用。
三 nfc過(guò)濾標(biāo)簽的設(shè)置
3-1 在Manifest添加權(quán)限:
在xml里添加nfc的使用權(quán)限
<uses-permission android:name="android.permission.NFC" />
這個(gè)是限制安裝權(quán)限,只給有nfc功能的手機(jī)安裝(可選)
<uses-feature android:name="android.hardware.nfc" android:required="true" />
3-2 nfc的過(guò)濾方式有以下:
ACTION_NDEF_DISCOVERED,
ACTION_TECH_DISCOVERED,
ACTION_TAG_DISCOVERED
三種。過(guò)濾器的作用是過(guò)濾出我們需要的標(biāo)簽類(lèi)型。這三種過(guò)濾方式可同時(shí)配置,可以比方成從上到下三層,只要是符合某一層過(guò)濾器要求的,過(guò)濾完就停止往下一層。
在Activity的filter里面添加對(duì)應(yīng)需要的權(quán)限:
ACTION_NDEF_DISCOVERED:,
<activity>
...
<intent-filter>
<action android:name="android.nfc.action.NDEF_DISCOVERED"/>
<category android:name="android.intent.category.DEFAULT"/>
<!-- 以下是常規(guī)的識(shí)別NFC芯片的配置 -->
<data android:mimeType="text/plain"/>
<!-- 以下是用來(lái)識(shí)別寫(xiě)入了網(wǎng)址的NFC芯片的配置,這兩項(xiàng)是Android 4.X原生系統(tǒng)自帶瀏覽器的配置 -->
<!-- <data android:scheme="http" />-->
<!-- <data android:scheme="https" />-->
</intent-filter>
...
</activity>
如果你的NFC卡片中的數(shù)據(jù)是純文本,那么在mimeType屬性中使用“text/plain”即可。另一種常見(jiàn)的使用場(chǎng)景是卡片中寫(xiě)有網(wǎng)址,在這種情況下,mimeType可以使用“http”或"https"
ACTION_TECH_DISCOVERED:
在<project-root>/res/xml(自己新建xml文件夾)下新建一個(gè)nfc_tech_filter.xml文件,添加進(jìn)你需要支持的標(biāo)簽類(lèi)型。(下面的配置項(xiàng)可多選)。下列示例是支持與NfcA和Ndef技術(shù)的NFC標(biāo)簽匹配。
<resources xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
<tech-list>
<resources >
<tech-list>
<!--<tech>android.nfc.tech.IsoDep</tech>-->
<tech>android.nfc.tech.NfcA</tech>
<!--<tech>android.nfc.tech.NfcB</tech>-->
<!--<tech>android.nfc.tech.NfcF</tech>-->
<!--<tech>android.nfc.tech.NfcV</tech>-->
<tech>android.nfc.tech.Ndef</tech>
<!--<tech>android.nfc.tech.NdefFormatable</tech>-->
<!--<tech>android.nfc.tech.MifareClassic</tech>-->
<!--<tech>android.nfc.tech.MifareUltralight</tech>-->
</tech-list>
</resources>
</tech-list>
</resources>
<activity>
...
<intent-filter>
<action android:name="android.nfc.action.TECH_DISCOVERED" />
</intent-filter>
<meta-data
android:name="android.nfc.action.TECH_DISCOVERED"
android:resource="@xml/nfc_tech_filter" />
<meta-data android:name="android.nfc.action.TECH_DISCOVERED"
android:resource="@xml/nfc_tech_filter" />
...
</activity>
ACTION_TAG_DISCOVERED,可以添加如下權(quán)限
<activity>
···
<intent-filter>
<action android:name="android.nfc.action.TAG_DISCOVERED" />
</intent-filter>
···
</activity>
3-3 識(shí)別標(biāo)簽的順序

四 nfc讀操作(我們讀取NEDF數(shù)據(jù),其他公交卡類(lèi)型的數(shù)據(jù)可以自行研究)
1 初始化nfc工具,判斷是否存在nfc和nfc是否打開(kāi)
2 感應(yīng)到nfc標(biāo)簽后,讀取解析對(duì)應(yīng)nfc類(lèi)型的標(biāo)簽數(shù)據(jù)
3 回傳顯示
代碼如下:
public class NfcActivity extends Activity {
private static final String TAG = "NfcActivity";
private TextView tvNFCMessage;
private PendingIntent mPendingIntent;
private NfcAdapter mNfcAdapter;
private Button btnClean;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_nfc);
Log.i(TAG, "onCreate: ");
btnClean = findViewById(R.id.btn_clean);
tvNFCMessage = findViewById(R.id.tv_show_nfc);
resolveIntent(getIntent());
//初始化nfc
mNfcAdapter = NfcAdapter.getDefaultAdapter(this);
mPendingIntent = PendingIntent.getActivity(this, 0,
new Intent(this, getClass()).addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP), 0);
if (mNfcAdapter == null) {
Toast.makeText(NfcActivity.this, "nfc is not available", Toast.LENGTH_SHORT).show();
finish();
return;
}
btnClean.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
tvNFCMessage.setText("");
}
});
}
@Override
protected void onResume() {
super.onResume();
Log.i(TAG, "onResume: ");
if (mNfcAdapter != null) { //有nfc功能
if (mNfcAdapter.isEnabled()) {
//nfc功能打開(kāi)了
//隱式啟動(dòng)
mNfcAdapter.enableForegroundDispatch(this, mPendingIntent, null, null);
} else {
Toast.makeText(NfcActivity.this, "請(qǐng)打開(kāi)nfc功能", Toast.LENGTH_SHORT).show();
}
}
}
@Override
protected void onNewIntent(Intent intent) {
super.onNewIntent(intent);
Log.i(TAG, "onNewIntent: ");
setIntent(intent);
if (mNfcAdapter != null) { //有nfc功能
if (mNfcAdapter.isEnabled()) {//nfc功能打開(kāi)了
resolveIntent(getIntent());
} else {
Toast.makeText(NfcActivity.this, "請(qǐng)打開(kāi)nfc功能", Toast.LENGTH_SHORT).show();
}
}
}
@Override
protected void onPause() {
super.onPause();
if (mNfcAdapter != null) {
mNfcAdapter.disableForegroundDispatch(this);
}
}
//初次判斷是什么類(lèi)型的NFC卡
private void resolveIntent(Intent intent) {
NdefMessage[] msgs = NfcUtil.getNdefMsg(intent); //重點(diǎn)功能,解析nfc標(biāo)簽中的數(shù)據(jù)
if (msgs == null) {
Toast.makeText(NfcActivity.this, "非NFC啟動(dòng)", Toast.LENGTH_SHORT).show();
} else {
setNFCMsgView(msgs);
}
}
/**
* 顯示掃描后的信息
*
* @param ndefMessages ndef數(shù)據(jù)
*/
@SuppressLint("SetTextI18n")
private void setNFCMsgView(NdefMessage[] ndefMessages) {
if (ndefMessages == null || ndefMessages.length == 0) {
return;
}
// tvNFCMessage.setText("Payload:" + new String(ndefMessages[0].getRecords()[0].getPayload()) + "\n");
Calendar calendar = Calendar.getInstance();
int hour = calendar.get(Calendar.HOUR_OF_DAY);
int minute = calendar.get(Calendar.MINUTE);
tvNFCMessage.append(hour + ":" + minute + "\n");
List<ParsedNdefRecord> records = NdefMessageParser.parse(ndefMessages[0]);
final int size = records.size();
for (int i = 0; i < size; i++) {
ParsedNdefRecord record = records.get(i);
tvNFCMessage.append(record.getViewText() + "\n");
}
}
}
解析不同類(lèi)型nfc類(lèi)型的數(shù)據(jù)的方法(重點(diǎn)方法):
//初次判斷是什么類(lèi)型的NFC卡
public static NdefMessage[] getNdefMsg(Intent intent) {
if (intent == null)
return null;
//nfc卡支持的格式
Tag tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
String[] temp = tag.getTechList();
for (String s : temp) {
Log.i(TAG, "resolveIntent tag: " + s);
}
String action = intent.getAction();
if (NfcAdapter.ACTION_NDEF_DISCOVERED.equals(action) ||
NfcAdapter.ACTION_TECH_DISCOVERED.equals(action) ||
NfcAdapter.ACTION_TAG_DISCOVERED.equals(action)) {
Parcelable[] rawMessage = intent.getParcelableArrayExtra(NfcAdapter.EXTRA_NDEF_MESSAGES);
NdefMessage[] ndefMessages;
// 判斷是哪種類(lèi)型的數(shù)據(jù) 默認(rèn)為NDEF格式
if (rawMessage != null) {
Log.i(TAG, "getNdefMsg: ndef格式 ");
ndefMessages = new NdefMessage[rawMessage.length];
for (int i = 0; i < rawMessage.length; i++) {
ndefMessages[i] = (NdefMessage) rawMessage[i];
}
} else {
//未知類(lèi)型 (公交卡類(lèi)型)
Log.i(TAG, "getNdefMsg: 未知類(lèi)型");
//對(duì)應(yīng)的解析操作,在Github上有
}
return ndefMessages;
}
return null;
}
最后,附上我的demo地址,歡迎大家學(xué)習(xí)下載,有什么問(wèn)題也歡迎找我討論:
https://github.com/younger96/NFCRead