8

JobIntentService每次推送通知进来时我都会启动一个来告诉服务去获取更多数据。当应用程序在前台时,一切正常。

当应用程序在后台并且有多个推送通知进来时,意图会排队并多次执行相同的意图给服务器带来不必要的压力,因为对服务器的第一次调用将获得它需要的所有信息进行另一个来自推送通知的排队意图是不必要的。

是否有取消或不向队列添加相同的意图或其他方式来防止对服务器的额外调用?

4

2 回答 2

3

所以首先是几个已经相似且知识渊博的问题:question1question2。也许您的问题甚至重复了类似的问题。

最干净的方法是在您的MyOwnJobIntentService extends JobIntentService.

里面有androidx.core.app.JobIntentService.java

final ArrayList<CompatWorkItem> mCompatQueue;

在你的MyOwnJobIntentService extends JobIntentService

if(mCompatQueue.isEmpty()){
  //only in this case enqueue new job   
}

但可惜mCompatQueue不是公场

10 分钟后,我们得到了工作解决方案 ->SingleJobIntentService一个 JobIntentService,如果它已经在工作,它将不会排队工作。

import android.content.Context;
import android.content.Intent;
import android.support.annotation.NonNull;
import android.support.v4.app.JobIntentService;
import android.util.Log;

public class SingleJobIntentService extends JobIntentService {

    private static final String TAG = "SingleJobIntentService";

    private Intent theOnlyJobIhave = null;

    public static void enqueue(Context context, Intent work) {
        Log.d(TAG, "enqueue: someone tries to add me work " + work.hashCode());
        JobIntentService.enqueueWork(
                context,
                SingleJobIntentService.class,
                SingleJobIntentService.class.hashCode(),
                work);
    }

    @Override
    protected void onHandleWork(@NonNull final Intent theWorkIgot) {
        Log.d(TAG, "onHandleWork: " + this.hashCode());
        if (theOnlyJobIhave == null) {
            theOnlyJobIhave = theWorkIgot;
            final int extraValue = theOnlyJobIhave.getIntExtra(MainActivity.KEY, -500);
            Log.d(TAG, "onHandleWork: " + extraValue);
            try {
                Thread.sleep(7000); //this simulates fetch to server
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        } else {
            Log.d(TAG, "onHandleWork I'm already busy, refuse to work >:(");
        }
        Log.d(TAG, "onHandleWork end");
    }
}

你可能想用一个简单的按钮来测试它:

import android.content.Intent;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;

public class MainActivity extends AppCompatActivity {

    public static final String KEY = "KEYKEY";

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

        findViewById(R.id.button).setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(final View v) {
                final Intent theIntent = new Intent();
                theIntent.putExtra(KEY, 666);
                SingleJobIntentService.enqueue(MainActivity.this, theIntent);
            }
        });
    }
}

注意:你必须小心线程,因为我给出的解决方案不是线程安全的。例如在androidx.core.app.JobIntentService.java触摸mCompatQueue中是同步的。 编辑:在考虑之后 - >因为onHandleWork从单线程调用 JobIntentService 没有线程问题并且解决方案是线程安全的!

于 2019-07-11T14:34:43.063 回答
-2

您可以对所有通知使用相同的通知 ID,这样它将覆盖旧通知并为您提供唯一具有最新意图的最新通知。

如果您有针对不同操作的多种类型的通知,您可以创建不同的通知组,提供仅适用于特殊类型操作的意图。

并且还要继续清除通知组,因此一旦单击其中一个通知,您就可以清除同一组通知。

于 2019-07-11T05:41:08.393 回答