【问题标题】:Starting app when MIFARE Classic is detect with Android NFC使用 Android NFC 检测到 MIFARE Classic 时启动应用程序
【发布时间】: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.htmlhttp://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


【解决方案1】:

您的问题是技术过滤器 XML 文件(您终于正确地发现了自己)。您最初使用的技术过滤器,

<?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>

没有任何意义。一个&lt;tech-filter&gt; 条目中的&lt;tech&gt; 条目使用逻辑AND 组合。

所以你需要有一个标签是Ndef NdefFormatable MifareClassic MifareUltralight。这是不可能的,原因有两个:

  1. NdefNdefFormatable 是互斥的。标签可以或者已经包含 NDEF 数据/空的 NDEF 消息 (->Ndef)或者它可能准备好使用 NDEF 消息进行格式化 (->@ 987654333@)。
  2. MifareClassicMifareUltralight 是互斥的。标签可以要么是 MIFARE Classic 标签 MIFARE Ultralight 标签,但不能同时是两者。

因此,例如,在包含 NDEF 消息 的标签上触发的适当技术过滤器是 MIFARE Classic 或 MIFARE Ultralight,如下所示:

<?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.MifareClassic</tech>
    </tech-list>
    <tech-list>
        <tech>android.nfc.tech.Ndef</tech>
        <tech>android.nfc.tech.MifareUltralight</tech>
    </tech-list>
</resources>

所有&lt;tech-list&gt; 条目都用逻辑OR 组合然后:(Ndef and MifareClassic) or Ndef and MifareUltralight)。

在您的情况下,您似乎对 NDEF 不太感兴趣,但对从 MIFARE 卡中获取任何东西更感兴趣。假设您所说的 MIFARE 是指 MIFARE Classic 卡(因为您说的是 1996 年 ;-)),您的技术过滤器应该如下所示:

<?xml version="1.0" encoding="utf-8"?>
<resources xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
    <tech-list>
        <tech>android.nfc.tech.MifareClassic</tech>
    </tech-list>
</resources>

但请注意,这仅适用于配备 NXP 芯片组的设备(MIFARE Classic 是 NXP 专有技术。虽然 NXP 将卡端许可给其他制造商,但他们不许可读卡器端。因此,只有1 NXP 的阅读器产品读取 MIFARE Classic。)。在具有 Broadcom 芯片组的设备上,MIFARE Classic 卡2 不会被检测为 MIFARE Classic。但由于 MIFARE Classic 使用标准 ISO 14443-3 Type A 防冲突和激活序列,这些设备通常3检测到NfcA 之类的卡:

<?xml version="1.0" encoding="utf-8"?>
<resources xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
    <tech-list>
        <tech>android.nfc.tech.NfcA</tech>
    </tech-list>
</resources>

请注意,使用带有MifareClassic 标签技术的第二个条目将是多余的,因为每个检测到 MIFARE Classic 的设备都会将其检测为MifareClassic NfcA


1) 该规则有例外,因为一些制造商不同意恩智浦的立场(或干脆忽略它?),即在阅读器端实施 MIFARE Classic 支持侵犯了他们的权利。

2) 只有 MIFARE 经典。这不适用于符合标准的 MIFARE 产品,例如 Ultralight、DESFire、NTAG 等。

3) 一些带有 Broadcom NFC 芯片组的三星设备,如 S4,是该规则的例外。在这些设备上,三星决定完全禁止 MIFARE Classic,而是显示“不支持标签”错误。正如他们声称的那样,为了改善用户体验,否则用户将不明白为什么他们不能将数据写入这些标签。或者正如解释的那样,让你的用户讨厌作为一个应用程序开发者,因为不能让你的应用程序在他们的手机上运行他们的标签。在ReTag app 或我的own app 上查看负面评论。

【讨论】:

  • 感谢 Michael 的非常好的解释……经过反复试验,我意识到技术过滤器问题。 Andorid NFC Basics and Advanced 文档对此主题并不清楚。他们没有详细解释技术过滤器,只是顺便说一句……我选择使用 NfcA,因为它可以同时处理 MIFARE 和 ULTRALIGHT。因为我只对标签的序列号感兴趣,所以我真的相信所有支持 NFC 的 ANDROID 手机都能读取它。我还使用 Foreground Dispatcher 来确保我的应用程序在前台时获得任何标签。再次感谢您。
  • 再次阅读文档后,他们提到了 AND 和 OR 语义......也许是我连续工作 12 小时后的思绪徘徊 :-)
  • Android NFC 文档确实解释了 ANDing 和 ORing 的概念,但它给出了两个无效示例 ;-)
  • @Loudenvier 您的代码对我帮助很大。我有一个问题:如果应用程序被最小化并且我读了一张卡片,它会加载应用程序 mainactivity 但显示 Hello world 并且不读取卡号
【解决方案2】:

经过大量搜索和挫折后,我开始尝试并意识到您无法过滤所有 Nfc 标记类型。其中一些与其他的不能很好地配合(例如 NfcA 和 MifareClassic)。我没有试图找出所有可能的无效组合,但仅通过 NfcA 过滤就可以解决问题(MifareClassic 和 Ultralight 也会被检测到,因为它们的行为类似于 NfcA)。

技术列表改为:

<resources xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
    <tech-list>
        <tech>android.nfc.tech.NfcA</tech>
    </tech-list>
</resources>

清单已更改为:

<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" />
            <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>

然后它起作用了!活动如期开展。

ANDROID 文档中没有提及这一点。我发现它大部分时间都需要参考它!但这就是生活……又一个不眠之夜(巴西凌晨 4 点),今天的问题少了一个。 :-)

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2021-10-12
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2016-06-05
    • 2014-12-28
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多