高级NFC(Advanced NFC)




这篇文档介绍了高级NFC主题,例如使用各种标签技术,允许一个应用在前台处理intent哪怕是其他应用过滤了同一个的前台分发。


与被支持的标签协作(Working with Supported Tag Technologies)


当安卓设备和NFC标签交互的时候,大多数时候你在标签上读、写的主要格式是NDEF。当设备扫描到NDEF数据,Android支持在NdefMessage中解析和传递消息,在可能的情况下。但是仍然有很多时候你扫描到的标签不包含NDEF数据或者NDEF数据不能被映射到一个MIME类型或URI。在这些情况下,你需要使用你自己的协议直接与标签通讯和读写信息(在原始字节)。Android通过android.nfc.tech包支持这些情形,在表1有具体描述。你可以使用getTechList()方法确定被支持的标签技术,用android.nfc.tech包下的类建立相应的TagTechnology对象。


表1. 支持的标签技术


类(Class)

描述(Description)

TagTechnology

所有标签技术类都必须实现的接口

NfcA

提供访问NFC-A (ISO 14443-3A)属性和I/O操作

NfcB

提供访问NFC-B (ISO 14443-3B) 属性和I/O操作

NfcF

提供访问NFC-F (JIS 6319-4)属性和I/O操作

NfcV

提供访问NFC-V (ISO 15693) 属性和I/O操作

IsoDep

提供访问ISO-DEP (ISO 14443-4) 属性和I/O操作

Ndef

提供访问被格式化的NFC标签上的NDEF数据和操作

NdefFormatable

提供可能被NDEF格式化的标签的格式化操作


表2. 可选的支持标签


类(Class)

描述(Description)

MifareClassic

提供访问MIFARE传统属性和I/O操作,如果安卓设备支持MIFARE.

MifareUltralight

提供访问MIFARE的Ultralight属性和I/O操作,如果安卓设备支持MIFARE.



使用标签技术和ACTION_TECH_DISCOVERED intent(Working with tag technologies and the ACTION_TECH_DISCOVERED intent)


当设备扫描到一个带有NDEF数据但不能被映射到一个MIME或者URI的标签,tag dispatch system会试着通过ACTION_TECH_DISCOVERED intent启动一个activity。当被扫描到带有non-NDEF数据的标签的时候ACTION_TECH_DISCOVERED也会被使用。这种保守策略使得当tag dispatch system不能解析你的标签的时候你仍然能正确处理标签上的数据。使用标签技术的基本步骤如下所示:


1. 过滤你想处理的用来说明标签技术的 ACTION_TECH_DISCOVERED intent。更多信息参阅 Filtering for NFC intents。一般来说,当一个NDEF消息 不能被映射到一个MIME类型或者URI,或者扫描到的标签不带有NDEF数据的时候, tag dispatch system会启动一个 ACTION_TECH_DISCOVERED intent。想看更多关于如何被决定的,请看 The Tag Dispatch System。


2. 当你的应用接收到一个intent,得到里面的Tag对象:


Tag tagFromIntent = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);


3. 得到一个TagTechnology的实例,通过调用android.nfc.tech包下的类的工厂方法:get。你在调用get工厂方法之前可以调用getTechList()列举被支持的标签。例如得到一个Tag中的MifareUltralight的实例,像下面这样:


MifareUltralight.get(intent.getParcelableExtra(NfcAdapter.EXTRA_TAG));



读写标签(Reading and writing to tags)


读写NFC标签包括从标签中得到intent和跟标签通讯。你必须定义你自己的协议栈来向标签读写数据。然后需要注意的是,当正确处理了一个标签你仍然可以读写NDEF数据。这取决于你如何组织事务。下面的例子展示了如何使用MIFARE Ultralight标签。


package com.example.android.nfc;import android.nfc.Tag;import android.nfc.tech.MifareUltralight;import android.util.Log;import java.io.IOException;import java.nio.charset.Charset;public class MifareUltralightTagTester {    private static final String TAG = MifareUltralightTagTester.class.getSimpleName();    public void writeTag(Tag tag, String tagText) {        MifareUltralight ultralight = MifareUltralight.get(tag);        try {            ultralight.connect();            ultralight.writePage(4, "abcd".getBytes(Charset.forName("US-ASCII")));            ultralight.writePage(5, "efgh".getBytes(Charset.forName("US-ASCII")));            ultralight.writePage(6, "ijkl".getBytes(Charset.forName("US-ASCII")));            ultralight.writePage(7, "mnop".getBytes(Charset.forName("US-ASCII")));        } catch (IOException e) {            Log.e(TAG, "IOException while closing MifareUltralight...", e);        } finally {            try {                ultralight.close();            } catch (IOException e) {                Log.e(TAG, "IOException while closing MifareUltralight...", e);            }        }    }    public String readTag(Tag tag) {        MifareUltralight mifare = MifareUltralight.get(tag);        try {            mifare.connect();            byte[] payload = mifare.readPages(4);            return new String(payload, Charset.forName("US-ASCII"));        } catch (IOException e) {            Log.e(TAG, "IOException while writing MifareUltralight            message...", e);        } finally {            if (mifare != null) {               try {                   mifare.close();               }               catch (IOException e) {                   Log.e(TAG, "Error closing tag...", e);               }            }        }        return null;    }}



使用前台发布系统(Using the Foreground Dispatch System)


前台发布系统允许activity拦截一个intent并且声明对处理同一个intent的比其他activity更高的优先权。使用这个系统涉及到构建一些使安卓系统能向你的应用发送合适的intent的数据结构。启用前台调度系统:


1. 添加如下代码到你activity的onCreate()方法中:


        1). 创建一个PendingIntent对象使得当标签被扫描的时候安卓系统可以用标签描述来填充它。


PendingIntent pendingIntent = PendingIntent.getActivity(    this, 0, new Intent(this, getClass()).addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP), 0);


intent 的你想过滤的标签。下面的代码片段处理了所有NDEF_DISCOVERED的MIME类型。 你应该只处理你需要的那些。


IntentFilter ndef = new IntentFilter(NfcAdapter.ACTION_NDEF_DISCOVERED);    try {        ndef.addDataType("*/*");    /* Handles all MIME based dispatches.                                       You should specify only the ones that you need. */    }    catch (MalformedMimeTypeException e) {        throw new RuntimeException("fail", e);    }   intentFiltersArray = new IntentFilter[] {ndef, };


        3). 创建你的应用程序想处理的标签技术的数组。调用Object.class.getName()方法来得到你想支持的技术的类。


techListsArray = new String[][] { new String[] { NfcF.class.getName() } };


onPause())和重新获得( onResume())焦点的时候添加逻辑来启用和禁用前台发布系统。 enableForegroundDispatch()必须在主线程中且activity在前台(在onReaume()中调用以保证效果)的时候调用。你也必须实现onNewIntent回调来处理从NFC标签中的数据。


public void onPause() {    super.onPause();    mAdapter.disableForegroundDispatch(this);}public void onResume() {    super.onResume();    mAdapter.enableForegroundDispatch(this, pendingIntent, intentFiltersArray, techListsArray);}public void onNewIntent(Intent intent) {    Tag tagFromIntent = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);    //do something with tagFromIntent}



想看完整的例子请到API Demos中查看ForegroundDispatch。