用于描述NDEF格式数据的两个重要的类
NdefMessage:描述NDEF格式的信息
NdefRecord:描述NDEF信息的一个信息段
NdefMessage和NdefRecord是Android NFC技术的核心类,无论读写NDEF格式的NFC标签,还是通过Android Beam技术传递Ndef格式的数据,都需要这两个类。
向NFC标签写入数据的步骤
获取Tag对象
Tag tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
判断NFC标签的数据类型(通过Ndef.get方法)
Ndef ndef = Ndef.get(tag);
写入数据
ndef.writeNdefMessage(ndefMessage);
示例:自动启动Android应用程序
最终效果:写入后,关闭程序,回到主页面,然后拿NFC标签靠近手机后盖上部,程序回自动运行。 实现代码如下所示。
import java.util.ArrayList;
import java.util.List; import android.app.ListActivity;
import android.content.Intent;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.os.Bundle;
import android.view.View;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.ArrayAdapter; /**
* LIST列表,显示所有的包。
* @author dr
*/
public class InstalledApplicationListActivity extends ListActivity implements
OnItemClickListener { private List<String> mPackages = new ArrayList<String>(); @Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState); // 获取所有程序包名称,并且循环显示出来。
PackageManager packageManager = getPackageManager();
List<PackageInfo> packageInfos = packageManager
.getInstalledPackages(PackageManager.GET_ACTIVITIES);
for (PackageInfo packageInfo : packageInfos) {
mPackages.add(packageInfo.applicationInfo.loadLabel(packageManager)
+ "\n" + packageInfo.packageName);
} ArrayAdapter<String> arrayAdapter = new ArrayAdapter<String>(this,
android.R.layout.simple_list_item_1, android.R.id.text1,
mPackages);
setListAdapter(arrayAdapter);
// 列表项,单击事件。
getListView().setOnItemClickListener(this);
} @Override
public void onItemClick(AdapterView<?> parent, View view, int position,
long id) {
Intent intent = new Intent();
// 把选中的包名传过去
intent.putExtra("package_name", mPackages.get(position));
setResult(1, intent);
finish(); } }
mport android.app.Activity;
import android.app.PendingIntent;
import android.content.Intent;
import android.nfc.NdefMessage;
import android.nfc.NdefRecord;
import android.nfc.NfcAdapter;
import android.nfc.Tag;
import android.nfc.tech.Ndef;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.Toast; public class RunApplicationActivity extends Activity { private Button mSelectAutoRunApplication;
private String mPackageName;
private NfcAdapter mNfcAdapter;
private PendingIntent mPendingIntent; @Override
protected void onCreate(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_auto_run_application); mSelectAutoRunApplication = (Button) findViewById(R.id.button_select_auto_run_application); mNfcAdapter = NfcAdapter.getDefaultAdapter(this);
// 一旦截获NFC消息后,通过PendingIntent来调用
mPendingIntent = PendingIntent.getActivity(this, 0, new Intent(this,
getClass()), 0); } public void onResume() { // 当窗口获得焦点时
super.onResume(); if (mNfcAdapter != null)
// (一旦截获NFC消息)优先级,优于所有的(处理NFC标签)窗口
mNfcAdapter.enableForegroundDispatch(this, mPendingIntent, null,
null);
} public void onPause() { //
super.onPause(); if (mNfcAdapter != null)
// 取消,把窗口恢复到正常状态。
mNfcAdapter.disableForegroundDispatch(this);
} public void onClick_SelectAutoRunApplication(View view) {
Intent intent = new Intent(this, InstalledApplicationListActivity.class);
startActivityForResult(intent, 0);
} /**
* 因为此Activity配置配件中设置成singleTop(第2次运行onCreate将不会创建新的窗口实例),
* 不能在onCreate中获取Intent传过来的TAG数据。 但是,会调用此方法,onNewIntent也是Activity里面的方法。
*/
public void onNewIntent(Intent intent) {
if (mPackageName == null)
return;
// 获得Tag。
Tag detectedTag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
// 写入标签。
writeNFCTag(detectedTag);
} public void writeNFCTag(Tag tag) {
if (tag == null) {
return;
}
NdefMessage ndefMessage = new NdefMessage(
new NdefRecord[] { NdefRecord
.createApplicationRecord(mPackageName) });
int size = ndefMessage.toByteArray().length;
try {
// 判断NFC标签的数据类型。
Ndef ndef = Ndef.get(tag);
if (ndef != null) {
ndef.connect(); // 建立连接 if (!ndef.isWritable()) { // 判断NFC标签是否可写。
return;
}
if (ndef.getMaxSize() < size) { // 最大尺寸<写入尺寸。
return;
}
// 写入数据。
ndef.writeNdefMessage(ndefMessage);
Toast.makeText(this, "ok", Toast.LENGTH_LONG).show();
} } catch (Exception e) {
// TODO: handle exception
}
} @Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (resultCode == 1) {
mSelectAutoRunApplication.setText(data.getExtras().getString(
"package_name"));
String temp = mSelectAutoRunApplication.getText().toString();
mPackageName = temp.substring(temp.indexOf("\n") + 1); } } }
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" > <Button
android:id="@+id/button_select_auto_run_application"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:onClick="onClick_SelectAutoRunApplication"
android:text="选择已安装的应用程序" /> <TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="5dp"
android:text="请将NFC标签或贴纸靠近手机背面"
android:textSize="16sp" /> <ImageView
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_margin="10dp"
android:src="@drawable/read_nfc_tag" /> </LinearLayout>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="cn.eoe.run.application"
android:versionCode="1"
android:versionName="1.0" > <uses-sdk
android:minSdkVersion="15"
android:targetSdkVersion="15" /> <uses-permission android:name="android.permission.NFC" /> <application
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name=".RunApplicationActivity"
android:label="@string/title_activity_auto_run_application"
android:launchMode="singleTop"
android:screenOrientation="portrait" >
<intent-filter>
<action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity
android:name=".InstalledApplicationListActivity"
android:label="@string/title_activity_installed_application_list"
android:screenOrientation="portrait" /> </application> </manifest>