我在应用程序内部使用Pushwoosh来接收推送通知。我正在使用最新版本的Pushwoosh库3.1.14。

我有这样的屏幕结构。

Login Activity -> Main Activity with multiple tabs.

因此,我正在MainActivity中实现与pushwoosh相关的逻辑。我想从“注销”上的注销中注销,然后返回“登录 Activity ”。

我的代码如下。我已经过滤掉了与Pushwoosh不相关的所有其他部分。坦率地说,此代码与Pushwoosh文档here中的代码完全相似。唯一的区别是在onLogout()方法中,我尝试从pushwoosh注销并返回LoginActivity。

TabbarActivity.java
@Override
protected void onCreate(Bundle savedInstanceState) {
   super.onCreate(savedInstanceState);

   //Pushwoosh Registration
   registerReceivers();
   PushManager pushManager = PushManager.getInstance(this);
   pushManager.setNotificationFactory(new PushNotificationFactory());
   try {
       pushManager.onStartup(this);
   } catch(Exception e) {}

   //Register for push!
   pushManager.registerForPushNotifications();
   checkMessage(getIntent());
}


@Override
protected void onNewIntent(Intent intent) {
    super.onNewIntent(intent);
    setIntent(intent);
    checkMessage(intent);
}

@Override
protected void onResume() {
    super.onResume();
    registerReceivers();
}

@Override
protected void onPause() {
    super.onPause();
    unregisterReceivers();
}

BroadcastReceiver mBroadcastReceiver = new BaseRegistrationReceiver() {
   @Override
   public void onRegisterActionReceive(Context context, Intent intent) {
       checkMessage(intent);
   }
};

private BroadcastReceiver mReceiver = new BasePushMessageReceiver() {
   @Override    protected void onMessageReceive(Intent intent) {
     //JSON_DATA_KEY contains JSON payload of push notification.
   }
};

public void registerReceivers() {
  IntentFilter intentFilter = new IntentFilter(
    getPackageName() + ".action.PUSH_MESSAGE_RECEIVE");
  registerReceiver(mReceiver, intentFilter,
    getPackageName() +".permission.C2D_MESSAGE", null);
  registerReceiver(mBroadcastReceiver, new IntentFilter(
    getPackageName() + "." + PushManager.REGISTER_BROAD_CAST_ACTION));
}

public void unregisterReceivers() {
  try {
    unregisterReceiver(mReceiver);
  } catch (Exception e) {
    e.printStackTrace();
  }

  try {
    unregisterReceiver(mBroadcastReceiver);
  } catch (Exception e) {
    e.printStackTrace();
  }
}

private void checkMessage(Intent intent) {
  if (null != intent) {
    if (intent.hasExtra(PushManager.REGISTER_EVENT)) {
       uploadPushTokenToServer(PushManager.getPushToken(this));
    }
    resetIntentValues();
  }
}

private void resetIntentValues() {
  Intent mainAppIntent = getIntent();
  if (mainAppIntent.hasExtra(PushManager.PUSH_RECEIVE_EVENT)) {
    mainAppIntent.removeExtra(PushManager.PUSH_RECEIVE_EVENT);
  } else if (mainAppIntent.hasExtra(PushManager.REGISTER_EVENT)) {
    mainAppIntent.removeExtra(PushManager.REGISTER_EVENT);
  } else if (mainAppIntent.hasExtra(PushManager.UNREGISTER_EVENT)) {
    mainAppIntent.removeExtra(PushManager.UNREGISTER_EVENT);
  } else if (mainAppIntent.hasExtra(PushManager.REGISTER_ERROR_EVENT)) {
    mainAppIntent.removeExtra(PushManager.REGISTER_ERROR_EVENT);
  } else if (mainAppIntent.hasExtra(PushManager.UNREGISTER_ERROR_EVENT)) {
    mainAppIntent.removeExtra(PushManager.UNREGISTER_ERROR_EVENT);
  }
  setIntent(mainAppIntent);
}

//Finally on logout
private void onLogout() {
   //other cleanup

   //pushwoosh
   PushManager.getInstance(this).unregisterForPushNotifications();

   //goback to login activity
}

我正在从服务器获得推送,没有任何问题。我面临的唯一问题是注销并返回LoginActivity后,TabbarActivity仍保留在内存中,而内存又保留了许多其他 fragment 和 View 。我尝试使用MAT进行调试,这就是显示的内容。
Class Name                                                                      | Ref. Objects | Shallow Heap | Ref. Shallow Heap | Retained Heap
--------------------------------------------------------------------------------------------------------------------------------------------------
com.pushwoosh.internal.request.RequestManager$1 @ 0x12f89ce0  Thread-1737 Thread|            1 |           88 |               360 |           536
'- val$context in.myproject.activities.TabbarActivity         @ 0x12d8ac40      |            1 |          360 |               360 |        18,520
--------------------------------------------------------------------------------------------------------------------------------------------------

我还使用LeakCanary工具进行了交叉检查,这也表明Pushwoosh坚持了我的 Activity 。

所以我的问题是,如何清理pushwoosh以避免 Activity 泄漏?

最佳答案

通过简短地查看,您引用的文档仅提供了一个示例,这些示例并不总是将api集成到具有其他 Activity 的功能齐全的应用程序中的最佳方法。据了解,通过使用getInstance,您正在尝试使用单例,但是怀疑这没有得到很好的管理。

我将控制在应用期间使用的PushManager实例。

问题可能是在类中(可能在程序的生命周期内)从范围创建了PushManager的多个实例,并在类中创建了pushManager的多个实例。这将导致泄漏。

我将使pushManager成为类变量,而不是两次使用PushManager.getInstance并考虑创建在应用程序使用期间使用的PushManager的静态实例,就像在整个应用程序中使用单个数据库实例一样。

在类里面:

PushManager pushManager;

并在oncreate中初始化
pushManager = PushManager.getInstance(this);


//Finally on logout
private void onLogout() {
    //other cleanup

    //pushwoosh
    // Here the first instance is left dangling.
    // PushManager.getInstance(this).unregisterForPushNotifications();

    pushManager..unregisterForPushNotifications();

    //goback to login activity
}

这样,您就可以为一个pushmanager实例清理资源。

要使用应用程序范围内的静态PushManager,请执行以下操作:
static PushManager pushManager;

初始化为:
pushManager = new PushManager(this.getApplicationContext());

When is a Singleton not a Singleton?

关于android - Pushwoosh泄漏了我的 Activity ,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/33755410/

10-11 00:13