【发布时间】:2014-05-07 23:55:49
【问题描述】:
我在使用 MIFARE 方面拥有丰富的经验(自 1996 年以来,在使用 GEMPLUS 制造的卡时)。我什至编写了低级代码来模拟 MIFARE 卡……但现在事情变得更简单、更高级别了,我无法让它与 Java、Android 和 Android Studio 一起使用!我觉得随着时间的推移我越来越笨了......
我要做的就是在检测到 MIFARE 卡时启动应用程序。我知道它可以完成,因为我在我的设备中使用了 NFC 卡信息应用程序,并且它在 MIFARE 卡存在的情况下正确启动。我已经卸载它以确保唯一的 NFC 应用程序是我自己的。我尝试遵循在 http://developer.android.com/guide/topics/connectivity/nfc/nfc.html 和 http://developer.android.com/guide/topics/connectivity/nfc/advanced-nfc.html 上找到的不太令人满意的文档。
问题是我的应用程序从未启动。代码非常简单,而且,AFAIK 应该可以工作......这是应用清单:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.nfctest.app" >
<uses-sdk android:minSdkVersion="10"/>
<uses-permission android:name="android.permission.NFC" />
<uses-feature android:name="android.hardware.nfc" android:required="true" />
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name="com.example.nfctest.app.MainActivity"
android:label="@string/app_name" >
<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"/>
<category android:name="android.intent.category.DEFAULT"/>
</intent-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/nfc_tech_filter" />
</activity>
</application>
</manifest>
它按 NDEF_DISCOVERED 过滤,因为 TECH_DISCOVERED 不能单独工作。
这里是 nfc_tech_filter 资源文件:
<?xml version="1.0" encoding="utf-8"?>
<resources xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
<tech-list>
<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>
这是活动代码:
package com.example.nfctest.app;
import android.content.Intent;
import android.nfc.NfcAdapter;
import android.support.v7.app.ActionBarActivity;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.TextView;
public class MainActivity extends ActionBarActivity {
TextView label;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
label = (TextView)findViewById(R.id.label);
}
@Override
public void onResume() {
super.onResume();
if (NfcAdapter.ACTION_NDEF_DISCOVERED.equals(getIntent().getAction())) {
label.setText("NDEF_DISCOVERED");
} else if (NfcAdapter.ACTION_TECH_DISCOVERED.equals(getIntent().getAction())) {
label.setText("TECH_DISCOVERED");
}
}
@Override
public void onNewIntent(Intent intent) {
label.setText("onNewIntent!!!");
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
getMenuInflater().inflate(R.menu.main, menu);
return true;
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
int id = item.getItemId();
if (id == R.id.action_settings) {
return true;
}
return super.onOptionsItemSelected(item);
}
}
活动布局非常简单:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
android:paddingBottom="@dimen/activity_vertical_margin"
tools:context="com.example.nfctest.app.MainActivity">
<TextView
android:id="@+id/label"
android:text="@string/hello_world"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
</RelativeLayout>
开始编写实际 NFC 代码所需的只是在检测到 TAG 时启动应用程序。文档让它听起来很简单,以至于我可以确定是我,再次愚蠢......
更新:
通过将活动代码更改为,我可以使用 foregroundDispatcher 正确读取标签 UID:
package com.example.nfctest.app;
import android.app.PendingIntent;
import android.content.Intent;
import android.content.IntentFilter;
import android.nfc.NfcAdapter;
import android.nfc.Tag;
import android.support.v7.app.ActionBarActivity;
import android.os.Bundle;
import android.widget.TextView;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
public class MainActivity extends ActionBarActivity {
TextView label;
IntentFilter[] filters;
String[][] techs;
PendingIntent pendingIntent;
NfcAdapter adapter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
label = (TextView)findViewById(R.id.label);
pendingIntent = PendingIntent.getActivity(
this, 0, new Intent(this, getClass()).addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP), 0);
IntentFilter mifare = new IntentFilter((NfcAdapter.ACTION_TECH_DISCOVERED));
filters = new IntentFilter[] { mifare };
techs = new String[][] { new String[] { NfcA.class.getName() } };
adapter = NfcAdapter.getDefaultAdapter(this);
}
public void onPause() {
super.onPause();
adapter.disableForegroundDispatch(this);
}
public void onResume() {
super.onResume();
adapter.enableForegroundDispatch(this, pendingIntent, filters, techs);
}
public void onNewIntent(Intent intent) {
Tag tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
byte[] id = tag.getId();
ByteBuffer wrapped = ByteBuffer.wrap(id);
wrapped.order(ByteOrder.LITTLE_ENDIAN);
int signedInt = wrapped.getInt();
long number = signedInt & 0xffffffffl;
label.setText("Tag detected: " + number);
}
}
我什至不需要在清单中设置任何意图过滤器就可以了,只需一个像这样的简单清单就可以了:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.nfctest.app" >
<uses-sdk android:minSdkVersion="10"/>
<uses-permission android:name="android.permission.NFC" />
<uses-feature android:name="android.hardware.nfc" android:required="true" />
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name="com.example.nfctest.app.MainActivity"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
使用此代码,当我的应用程序在前台时,我可以读取 MIFARE 卡序列号,但我仍然无法进行设置,以便 ANDROID 在后台启动我的活动...
当检测到标签(MIFARE 或 NOT)时,我仍然不知道如何让 ANDROID 启动我的应用程序/活动。
【问题讨论】:
-
您使用什么设备进行测试?它会支持该标签吗?我似乎记得 MiFARE 标签有点不合规格,因此仅适用于某些 NFC 芯片。
-
我已经确定设备可以通过使用另一个应用程序(NFC 标签信息)测试来读取 MIFARE 卡。该设备是摩托罗拉 Razr i...
-
我也相信即使硬件不完全支持MIFARE,至少它可以检测到它并读取它的UID(AFAIK所有的NFC阅读器都可以读取一个MIFARE“标签id”/序列号)
-
@CommonsWare 现在我 100% 确定所有支持 NFC 的 ANDROID 设备至少可以读取 MIFARE UID...在我的更新中,我能够使用foregroundDispatcher 读取卡并设置使用 NfcA 而不是 MIFARE 的“技术”数组,它可以工作......现在剩下的就是让 ANDROID 在它不在前台时启动我的活动。
-
@Loudenvier 三星的几款基于 Broadcom NFC 芯片组的 Android 设备(例如 S4)无法读取(至少 4.4 之前的 Android 版本是这种情况,请不要不知道自那以后是否发生了变化)MIFARE Classic 卡(甚至没有 UID)。然而,这是三星的软件设计决定。带有 Broadcom 芯片组的设备可以正常读取 UID(因为 MF Classic 的防冲突等同于 ISO 14443-3)。另请注意,只有 MF Classic(由于其非 NfcA 框架)和 MF Classic-legacy 模式下的 MF Plus 受到影响,但 MF Ultralight、DESFire 等不受影响。
标签: android android-intent nfc mifare