3

我猜你们中的大多数人在看到这个标题时都会脸色苍白,并且认为我们以前见过这个我同意,但我已经尝试了一切来解决这个问题,但它不会补救。所以这里。

我的应用程序可以从主屏幕(图标)启动,一切正常,它会扫描 NFC 标签并将相关数据发送到服务器。但是,我还想要从 NFC 标签启动应用程序,这没有问题。然而!当应用程序通过 NFC 标签启动时,您然后移除手机并将其重新呈现给标签,然后它应该读取标签并将数据发送到服务器。与通过图标启动应用程序的行为完全相同,但是这次当您扫描 NFC 标签时,将启动应用程序的实例,最终将当前实例踢下堆栈,即忙于读取的实例版本标签,所以我得到一个 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();
}
4

2 回答 2

4

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

        android:launchMode="singleTask"
于 2013-09-04T11:29:37.430 回答
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先打电话。

于 2013-09-25T18:03:03.313 回答