7

常時インターネット接続が必要なアプリを開発中です。インターネット接続が存在しない場合、ユーザーをアプリからログアウトさせます (ログイン画面に移動します)。ネットワーク接続を検出するネットワーク レシーバー クラスがあります。このクラスで、スタック上のアクティビティを終了するか、新しいログイン アクティビティを開始して履歴スタック全体を削除するようにします。

問題は、レシーバー クラス内からフォアグラウンド アクティビティを終了できないことと、ネットワーク障害が発生したときにユーザーがどのアクティビティにいるのかを知る方法がないことです。また、このクラスから新しいログイン アクティビティを開始している場合、ユーザーが「戻る」を押すと (ネットワークに再接続すると) アプリに戻りますが、アプリはログインせずにクラッシュします。

myIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK || FLAG_ACTIVITY_CLEAR_TOP);NetworkStateReceiver クラスから新しいログイン アクティビティを開始するときに使用してみました。私の理解では、これは私が開始したアクティビティ(ログイン)だけの新しいタスクを作成しますが、他のすべてのアクティビティを含む古いタスクはそのまま残ります。

だから私は必要です:

-クラスからフォアグラウンド アクティビティを終了する方法

-または、クラスから新しいアクティビティを開始し、アクティビティ スタックを空にする方法

価値のあるレシーバーコードは次のとおりです。

public class NetworkStateReceiver extends BroadcastReceiver{
public void onReceive(Context context, Intent intent) {
//     super.onReceive(context, intent);
 Log.d("app","Network connectivity change");
 if(intent.getExtras()!=null) {
     Login.apiKey = null;
    NetworkInfo ni=(NetworkInfo) intent.getExtras().get(ConnectivityManager.EXTRA_NETWORK_INFO);
    if(ni!=null && ni.getState()==NetworkInfo.State.CONNECTED) {
        Log.i("app","Network "+ni.getTypeName()+" connected");
    }
 }
 if(intent.getExtras().getBoolean(ConnectivityManager.EXTRA_NO_CONNECTIVITY,Boolean.FALSE) && !Login.loginActive) {
        Log.d("app","There's no network connectivity");

        Toast.makeText(context, "No internet connection. Logging out", Toast.LENGTH_LONG).show();
        //logout
        Receiver.engine(context).halt();
        Receiver.mSipdroidEngine = null;
        Receiver.reRegister(0);
        new Thread(ChatList.runnable).interrupt();
        ChatList.buddyList.clear();
        Login.apiKey = null;
        Log.i("Logout", "Logged out!");
        Login.loggedOut = true;


        Intent myIntent = new Intent(context, Login.class);
        myIntent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_NEW_TASK);
//          myIntent.addFlags(Intent.FLAG_ACTIVITY_PREVIOUS_IS_TOP);
        context.startActivity(myIntent);



     }
   }
}

解決策: すべてのアクティビティからレシーバーへの参照を渡しました

//random user_activity
    @Override
protected void onPause() {
    super.onPause();
    NetworkStateReceiver.curActivity = null;
}


@Override
protected void onResume() {
    super.onResume();
    NetworkStateReceiver.curActivity = user_activity.this; //edited : getParent() doesn't always work
}

および のネットワーク受信機onReceive():

            if(curActivity != null)
        {
            curActivity.finish();
   }
4

1 に答える 1

3

1 つの方法は、レシーバーに現在のアクティビティへの参照を渡すことです (たとえば、onResume? で)。ただし、必ずヌルにしてください。そうしonPauseないと、見苦しいメモリ リークが発生します。次に、ログイン アクティビティを開始する前に、(null の場合は何もしない) ことができますonReceivecurActivity.finish()

于 2013-02-15T14:25:08.177 に答える