NFC科技代表近场通信,通常可以在 NFC论坛上找到 NFC的详细信息。下面我们将分析NFC 的一些基础知识,并且我们还将描述如何在安卓系统中实现一个利用 NFC的 APP(应用程序)。
如果你想做 NFC 实验,那么你在几个网站上花费几欧元就能买到NFC 。
NFC 可以用于不同的情景:在家中时,我们可以使用它来打开 WiFi或者执行一些任务操作等等。
我们将注意力集中在 NDEF数据,它是一种特殊类型的 NFC标记。不过,在使用 NFC之前,我们必须遵循一些基本步骤。
NFC 过滤器
当我们使用 NFC标记时,我们要做的第一件事就是,当我们接近一个 NFC标记的时候,我们的 APP 可以收到通知。为了实现这个目的,我们需要使用一个意图过滤器。安卓SDK中提供了三种不同的过滤器,我们可以以不同的优先级来使用它们:
ACTION_NDEF_DISCOVERED
ACTION_TECH_DISCOVERED
ACTION_TAG_DISCOVERED
我们重点关注 ACTION_NDEF_DISCOVERED,它拥有最高的优先级。正如前面所说的,我们的目的是,当智能手机接近一个 NFC 标记时能够收到通知,并且如果我们只安装了这一个 APP,并且它能够处理这个NFC标记,那么我们希望该APP能够立即开始处理。为了实现这一点,我们在Manifest.xml中注册过滤器:
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.survivingwithandroid.nfc" >
....
<intent-filter>
<action android:name="android.nfc.action.NDEF_DISCOVERED" />
<category android:name="android.intent.category.DEFAULT" />
<data android:mimeType="text/plain"/>
</intent-filter>
<manifest>
在第6行,我们注册了 APP,以使得它可以收到 ACTION_NDEF_DISCOVERED通知。我们使用不同类型的过滤器,在这个例子中(第 8 行)我们使用了 mine 类型。换句话说,当发现一个 NFC标记NDEF并且它的 mine类型为“text/plain”时,我们的 APP将会启动。其实,我们可以使用几种 mine 类型来过滤,而不仅仅是 text/plain。此外,我们还能以协议或一个字符串模式来使用其他类型的过滤器(如 android:scheme)来过滤。
NFC 前台调度
如果我们的 APP 没处于前台,就会开始意图过滤。如果我们的应用程序运行在前台,那么当我们的智能手机移动到一个NFC标记时,它将不会收到通知。在这种情况下,我们必须使用一种称为NFC前台调度的技术。第一步是,在我们的代码中定义意图过滤器(正如我们在 Manifest.xml 中做的那样):
@Override
protected void onCreate(Bundle savedInstanceState) {
...
Intent nfcIntent = new Intent(this, getClass());
nfcIntent.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);
nfcPendingIntent =
PendingIntent.getActivity(this, 0, nfcIntent, 0);
IntentFilter tagIntentFilter =
new IntentFilter(NfcAdapter.ACTION_NDEF_DISCOVERED);
try {
tagIntentFilter.addDataType("text/plain");
intentFiltersArray = new IntentFilter[]{tagIntentFilter};
}
catch (Throwable t) {
t.printStackTrace();
}
}
现在,我们必须注册我们的过滤器,在 onResume方法中以这种方式实现它:
protected void onResume() {
super.onResume();
nfcAdpt.enableForegroundDispatch(
this,
nfcPendingIntent,
intentFiltersArray,
null);
handleIntent(getIntent());
}
此外,也应该记住,只要APP退回到后台,我们就应该关闭前台调度,而要实现这一点,最好的地方是在onPause方法中。
@Override
protected void onPause() {
super.onPause();
nfcAdpt.disableForegroundDispatch(this);
}
其中,nfcAdpt 是NFC适配器。
使用 NFCAdapter处理NFC
一旦我们创建了过滤器,我们必须与智能手机中的 NFC 组件交互。为此,我们使用安卓 SDK 提供的 NfcAdapter。使用这个类,我们能够检测智能手机是否支持NFC,或者NFC功能是否开启或关闭。
@Override
protected void onCreate(Bundle savedInstanceState) {
...
nfcAdpt = NfcAdapter.getDefaultAdapter(this);
// Check if the smartphone has NFC
if (nfcAdpt == null) {
Toast.makeText(this, "NFC not supported", Toast.LENGTH_LONG).show();
finish();
}
// Check if NFC is enabled
if (!nfcAdpt.isEnabled()) {
Toast.makeText(this, "Enable NFC before using the app", Toast.LENGTH_LONG).show();
}
}
NFC数据:负载
既然我们知道了如何处理NFC标记,接着我们就想读取标记内容。在NFC说明文档中,定义了几种程序开发类型的内容:
NFC Forum well-known type
Media-type
Absolute URI
NFC Forum external type
每种类型都有它自己的有效负载。一般来说,一个NFC NDEF数据由一条信息组成,一条信息可以包含一个或多个记录,每个记录由一个头和一个负载(真正的信息)组成。现在,如果我们想读取NFC NDEF标记中的数据,我们可以使用下面的代码:
@Override
public void onNewIntent(Intent intent) {
Log.d("Nfc", "New intent");
getTag(intent);
}
private void getTag(Intent i) {
if (i == null)
return ;
String type = i.getType();
String action = i.getAction();
List dataList = new ArrayList();
if (NfcAdapter.ACTION_NDEF_DISCOVERED.equals(action)) {
Log.d("Nfc", "Action NDEF Found");
Parcelable[] parcs = i.getParcelableArrayExtra(NfcAdapter.EXTRA_NDEF_MESSAGES);
for (Parcelable p : parcs) {
recNumberTxt.setText(String.valueOf(numRec));
NdefRecord[] records = msg.getRecords();
for (NdefRecord record: records) {
short tnf = record.getTnf();
// Here we handle the payload
}
}
}
}