您的位置:首页 > 移动开发 > Android开发

Android NFC :ApiDemos->NFC->Fore…

2014-12-16 20:45 337 查看
本例参考ApiDemos中NFC的ForegoundDispatch来介绍编写Android NFC
的基本步骤,因为手边只有MifareClassic 类型的Tag
,需要对ForegoundDispatch的代码做些修改来检测MifareClassic 的类型的NFC
Tag,读写其他类型的NFC Tag的基本步骤是一致的。
1. 在Android
manifest 文件中申明和NFC相关的权限和功能选项:
权限申明:
<uses-permission
android:name=”android.permission.NFC” />
最低版本要求,NFC是指Android2.3 (Level 10)
才开始支持的,因此最低版本要求必须指定为10.
<uses-sdk
android:minSdkVersion=”10″/>
如果需要在Android Market上发布,需要指定手机支持NFC
功能。
<uses-feature
android:name=”android.hardware.nfc” android:required=”true”
/>
为Activity申明它支持处理NFC Tag
比如我们的示例Activity 在Manifest
的申明如下:
<activity
android:name=”.NFCDemoActivity”

android:label=”@string/app_name”

android:launchMode=”singleTop”>

<intent-filter>

<action android:name=”android.intent.action.MAIN”
/>

<category
android:name=”android.intent.category.LAUNCHER”
/>

</intent-filter>

<intent-filter>

<action
android:name=”android.nfc.action.NDEF_DISCOVERED”/>

<data android:mimeType=”text/plain”
/>

</intent-filter>

<intent-filter>

<action

android:name=”android.nfc.action.TAG_DISCOVERED”

>

</action>

<category

android:name=”android.intent.category.DEFAULT”

>

</category>

</intent-filter>

<!– Add a technology filter
–>

<intent-filter>

<action
android:name=”android.nfc.action.TECH_DISCOVERED”
/>

</intent-filter>
<meta-data
android:name=”android.nfc.action.TECH_DISCOVERED”

android:resource=”@xml/filter_nfc”

/>

</activity>
三种Activity NDEF_DISCOVERED
,TECH_DISCOVERED,TAG_DISCOVERED 指明的先后顺序非常重要, 当Android设备检测到有NFC
Tag靠近时,会根据Action申明的顺序给对应的Activity 发送含NFC消息的 Intent.
2. Android NFC 消息发送机制
当Android设备检测到有NFC
Tag时,理想的行为是触发最合适的Activity来处理检测到的Tag,这是因为NFC通常是在非常近的距离才起作用(<4m)
,如果此时需要用户来选择合适的应用来处理Tag,很容易断开与Tag之间的通信。因此你需要选择合适的Intent filter
只处理你想读写的Tag类型。
Android系统支持两种NFC消息发送机制:Intent
发送机制和前台Activity 消息发送机制。
Intent
发送机制
当系统检测到Tag时,Android系统提供manifest
中定义的Intent filter
来选择合适的Activity来处理对应的Tag,当有多个Activity可以处理对应的Tag类型时,则会显示Activity选择窗口由用户选择:


NFC :ApiDemos->NFC->ForegoundDispatch" TITLE="Android NFC :ApiDemos->NFC->ForegoundDispatch" />前台Activity
消息发送机制
允许一个在前台运行的Activity在读写NFC Tag
具有优先权,此时如果Android检测到有NFC Tag
,如果前台允许的Activity可以处理该种类型的Tag则该Activity具有优先权,而不出现Activity 选择窗口。
这两种方法基本上都是使用Intent-filter
来指明Activity可以处理的Tag类型,一个是使用Android的Manifest 来说明,一个是通过代码来申明。
下图显示当Android检测到Tag,消息发送的优先级:


NFC :ApiDemos->NFC->ForegoundDispatch" TITLE="Android NFC :ApiDemos->NFC->ForegoundDispatch" />
本例 NFCDemoActivity
支持两种NFC消息发送机制,上面的XML指明了Intent 消息发送机制,其中
<meta-data
android:name=”android.nfc.action.TECH_DISCOVERED”

android:resource=”@xml/filter_nfc”

/>
的filter_nfc 指明了支持处理的NFC
Tag类型,filter_nfc.xml 定义如下:
<resources
xmlns:xliff=”urn:oasis:names:tc:xliff:document:1.2″>

<!– capture anything using NfcF
–>

<tech-list>

<tech>android.nfc.tech.NfcA</tech>

<tech>android.nfc.tech.MifareClassic</tech>

<tech>android.nfc.tech.MifareUltralight</tech>

</tech-list>

</resources>
因为我只有MifareClassic
类型的Tag,所以只定义了MifareClassic相关的Tag类型,如果你可以处理所有Android支持的NFC类型,可以定义为:
<resources xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">    <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>


有了这个Manifest中的申明,当Android检测到有Tag时,会显示Activity选择窗口,如上图中的Reading
Example。

当NFCDemoActiviy在前台运行时,我们希望只有它来处理Mifare
类型的Tag,此时可以使用前台消息发送机制,下面的代码基本和ApiDemos中的NFC示例类似:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65
publicclassNFCDemoActivity extendsActivity {

privateNfcAdapter mAdapter;

privatePendingIntent mPendingIntent;

privateIntentFilter[] mFilters;

privateString[][] mTechLists;

privateTextView mText;

privateintmCount = 0;

@Override

publicvoidonCreate(Bundle savedState) {

super.onCreate(savedState);

setContentView(R.layout.foreground_dispatch);

mText = (TextView) findViewById(R.id.text);

mText.setText("Scan a tag");

mAdapter =
NfcAdapter.getDefaultAdapter(this);

// Create a generic PendingIntent that will be
deliver

// to this activity. The NFC stack

// will fill in the intent with the details of
the

//discovered tag before delivering to

// this activity.

mPendingIntent = PendingIntent.getActivity(this,
0,

newIntent(this,

getClass()).addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP),
0);

// Setup an intent filter for all MIME based
dispatches

IntentFilter ndef

=
newIntentFilter(NfcAdapter.ACTION_TECH_DISCOVERED);

try{

ndef.addDataType("*/*");

} catch(MalformedMimeTypeException e) {

thrownewRuntimeException("fail", e);

}

mFilters = newIntentFilter[] {

ndef,

};

// Setup a tech list for all MifareClassic
tags

mTechLists

= newString[][] { newString[] {
MifareClassic.class.getName() } };

}

@Override

publicvoidonResume() {

super.onResume();

mAdapter.enableForegroundDispatch(this,

mPendingIntent,
mFilters, mTechLists);

}

@Override

publicvoidonNewIntent(Intent intent) {

Log.i("Foreground dispatch",

"Discovered
tag with intent: "+ intent);

mText.setText("Discovered tag "+

++mCount
+ " with intent: "+ intent);

}

@Override

publicvoidonPause() {

super.onPause();

mAdapter.disableForegroundDispatch(this);

}

}

只改了一行,将处理NfcF类型的Tag
改为处理MifareClassic 类型的NFC Tag。
mTechLists = new
String[][] { new String[] { MifareClassic.class.getName() }
};
运行该示例,每靠近一次Tag,计数加1.


NFC :ApiDemos->NFC->ForegoundDispatch" TITLE="Android NFC :ApiDemos->NFC->ForegoundDispatch" />
本例下载

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: