Android NFC活动正在重新启动,即使它位于堆栈顶部

时间:2013-09-04 11:17:03

标签: android android-intent nfc flags android-pendingintent

当你看到这个标题的时候,我猜大多数人都面对着,并且认为我们之前已经看到了这个,我同意,但我已经尝试了一切来解决这个问题并且它不会补救。所以这里。

我的应用程序可以从主屏幕(图标)启动,一切正常,可以扫描NFC标签,并将相关数据发送到服务器。 然而,我也想要的是从NFC标签启动应用程序,它没有问题。 然而!当应用程序通过NFC标签启动时,您将移除手机并将其重新呈现给应该的标签,然后读取标签并将数据发送到服务器。与通过图标启动应用程序的行为完全相同,但是当您扫描NFC标签时,此时会启动应用程序的 new 实例,最终将当前实例从堆栈中踢出,版本为正在忙着读取标签的实例,所以我得到一个ANR。

简而言之:
从主屏幕 - >扫描标签 - >应用程序启动。
然后你重新扫描标签 - >
应用程序的当前实例消失,由新实例替换。 我希望以前的实例保持原样并且可以工作。

此时我的一些猜测是: - NFC未正确注册到应用程序,因此操作系统执行默认行为 - 通过NFC标签启动时没有正确设置标志,告诉操作系统不需要重新启动活动。

无论如何这里是代码:

<uses-permission android:name="android.permission.NFC"/>

<intent-filter>
            <action android:name="android.nfc.action.NDEF_DISCOVERED" />
            <category android:name="android.intent.category.DEFAULT" />
            <data android:scheme="http"android:host="@string/URL_for_intent_filtering" android:pathPrefix="/id/" />

//Procedure of NFC startup called from OnCreate
private void startNFC()
{
    mNfcAdapter = NfcAdapter.getDefaultAdapter(this);
    if(! mNfcAdapter.isEnabled())//check the NFC is switched on
    {
        mMainActivityHandler.sendEmptyMessage(NFC_DISABLED);
    }

    //its ok to carry on and instantiate the NFC even though its not enabled.
    if(mNfcAdapter != null)
    {
        int requestID = (int) System.currentTimeMillis();
        mPendingIntent = PendingIntent.getActivity(this, requestID, new Intent(this, getClass()).addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP), PendingIntent.FLAG_UPDATE_CURRENT);

        IntentFilter intentF = new IntentFilter(NfcAdapter.ACTION_NDEF_DISCOVERED);
        intentF.addDataScheme("http");

        intentfilter = new IntentFilter[] {intentF};



        if(DEBUG) Log.i("DEBUG","+++++++++++++++++++++++++++++++++++++++++++++++ NFC HAS BEEN STARTED");

    }else
    {
        //we have a problem with the NFC adapter probably because the device doesn't have one.
        if(DEBUG) Log.i("DEBUG","+++++++++++++++++++++++++++++++++++++++++++++++ NFC HAS FAILED");
        mMainActivityHandler.sendEmptyMessage(NFC_FAILED);
    }
}

@Override
protected void onPause()
{
    if(DEBUG) Log.i("MA","+++++++++++++++++++++++++++++++++++++++ ON PAUSE");
    try{
        mNfcAdapter.disableForegroundDispatch(this);
    }catch(Exception e)
    {

    }
    super.onPause();
}

//and here we should reinstate the NFC adapter
@Override
protected void onResume()
{
    if(DEBUG) Log.i("MA","+++++++++++++++++++++++++++++++++++++++ ON RESUME");

    //setupDataListener(true);
    setupServerComms(getApplicationContext());
    //mNfcAdapter.enableForegroundDispatch(this,mPendingIntent,intentfilter,null);
    mNfcAdapter.enableForegroundDispatch(this,mPendingIntent,null,null);
    super.onResume();
}

2 个答案:

答案 0 :(得分:4)

只需将AndroidManifest.xml中的这一行添加到您的Activity中,就可能已经完成了。

        android:launchMode="singleTask"

答案 1 :(得分:1)

那么你想要的是让应用程序捕获意图然后阅读信息并用它做正确的事情?那么你有一个待定的意图,但你的设置似乎与我通常做的有点不同。下面是一个示例,说明我如何使用挂起的意图来捕获标记数据(从那里你可以做任何你想做的事情,比如发送到服务器):

Class example{

NfcAdapter adapter;
Tag motor;
String FileName;
PendingIntent detectTag;
IntentFilter NFC;
IntentFilter [] Catcher;
String [][] TechList;
TextView readToField;

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_my_activity);

// Catch NFC detects
adapter = NfcAdapter.getDefaultAdapter(this);
detectTag = PendingIntent.getActivity(this,0,new Intent(this, getClass()).addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP), 0);
NFC = new IntentFilter(NfcAdapter.ACTION_TAG_DISCOVERED);
Catcher = new IntentFilter[] {NFC,};
TechList = new String [][]{new String[]{android.nfc.tech.NfcV.class.getName()}};
}

public void onNewIntent(Intent intent)
{
    super.onNewIntent(intent);
    motor = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
    Bridge bridge = (Bridge) getApplication();
    bridge.setCurrentTag(motor);
}

public void onPause()
{
    super.onPause();
    adapter.disableForegroundDispatch(this);
}

public void onResume()
{
    super.onResume();
    adapter.enableForegroundDispatch(this,detectTag ,Catcher , TechList);
}
}

Bridge对象只是我用来保存标记信息的类,所以我可以将信息发送到其他(幕后)类的标记。

最后,我始终被告知,对于onResumeonPause,您应首先致电super.method