0

我需要编写一个服务,它将每 30 秒更新一次 MainActivity 中的列表。我将 MVVM 与 ViewModel 和 LiveData 一起使用,因此我的 Service 类如下所示:

public class ArticleJobService extends JobService {

    public static String TAG = "ArticleJobService";

    private Context context = this;

    @Override
    public boolean onStartJob(JobParameters jobParameters) {
        Log.d(TAG, "onStartJob");
        MainActivity.PAGE_NUMBER++;
        LiveData<List<Article>> liveArticles = ArticleRepository.getInstance(getApplication()).getArticles(MainActivity.PAGE_NUMBER);

        liveArticles.observeForever(new Observer<List<Article>>() {
            @Override
            public void onChanged(@Nullable List<Article> articles) {
                Log.d(TAG, "onStartJob - onChanged!!!!!!");
                liveArticles.removeObserver(this);
                NotificationUtils.showNotification(context, articles.get(0).getSectionName(), articles.get(0).getWebTitle());
                jobFinished(jobParameters, true);
            }
        });
        return true;
    }
}

我的通知类:

public static void showNotification(Context context, String section, String title) {
    PendingIntent contentPendingIntent = PendingIntent.getActivity
            (context, REQUEST_CODE, new Intent(context, MainActivity.class),
                    PendingIntent.FLAG_UPDATE_CURRENT);

    NotificationManager manager =
            (NotificationManager) context.getSystemService(NOTIFICATION_SERVICE);


    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
        manager.createNotificationChannel(createNotificationChannel(context));
    }

    NotificationCompat.Builder builder = new NotificationCompat.Builder(context, CHANNEL_ID)
            .setContentTitle(section)
            .setContentText(title)
            .setContentIntent(contentPendingIntent)
            .setSmallIcon(R.drawable.app_icon)
            .setPriority(NotificationCompat.PRIORITY_HIGH)
            .setDefaults(NotificationCompat.DEFAULT_ALL)
            .setAutoCancel(true);

    manager.notify(0, builder.build());

}

当 JobService 中的 Onchanged 起作用时,我会得到列表并显示通知。通知会打开 MainActivity,它会像往常一样对 api 进行新的调用。为了让 MainActivity 显示我从服务中获得的列表,我必须进行哪些更改???我真的不能把它绑在一起。我听说过 IPC,但不会那样做,我想要一些我确实存在但我不知道的更简单的做法。另外,有两种情况:通知来了并且 MainActivity 已打开,应用程序已打开但 MainActivity 不在前台,应用程序在后台或关闭。我应该如何处理这些情况?

另请参阅 MainActivity onCreate 中的一段代码:

 mArticleViewModel = ViewModelProviders.of(this).get(ArticleViewModel.class);
    mArticleViewModel.getArticleList(PAGE_NUMBER).observe(this, articles -> {
        Log.d(TAG, "List<Result> onChanged!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!");
        mProgressBar.setVisibility(View.GONE);
        mProgressBarMain.setVisibility(View.GONE);
        mIsLoading = false;
        mArticles = articles;

请提供此任务的最佳实践,我知道这很常见,我只是第一次这样做,使用 LiveData 使它变得更加复杂。

这里也是存储库代码:

public static ArticleRepository getInstance(Application application){
        if(INSTANCE == null){
            return  new ArticleRepository(application);
        }

        return INSTANCE;
    }

    private ArticleRepository(Application application) {
        Log.d(TAG, "ArticleRepository constructor");
        mContext = application;
        mArticles = new MutableLiveData<>();
        ArticleRoomDatabase db = ArticleRoomDatabase.getInstance(application);
        mArticleDao = db.articleDao();
    }

    public LiveData<List<Article>> getArticles(int page) {
        Log.d(TAG, "getArticles");
        if (NetworkUtils.isOnline(mContext)) {
            Log.d(TAG, "isOnline");
            mArticles = loadFromNetwork(page);
        } else {
            Log.d(TAG, "is NOT Online");
            mArticles = loadFromDB(page);
        }
    }
4

1 回答 1

1

你有这个问题,特别是因为你的存储库实现不正确。

public LiveData<List<Article>> getArticles(int page) {
    Log.d(TAG, "getArticles");
    if (NetworkUtils.isOnline(mContext)) {
        Log.d(TAG, "isOnline");
        mArticles = loadFromNetwork(page);
    } else {
        Log.d(TAG, "is NOT Online");
        mArticles = loadFromDB(page);
    }
}

如果您检查NetworkBoundResource的代码,诀窍是您有一个 LiveData,它结合了从网络加载和从数据库加载的能力。

在您的情况下,只要您有网络访问权限,您就会替换数据库的自动更新查询结果 - 这就是您无法更新 MainActivity 的原因。

最简单的方法(不使用 MediatorLiveData)是在 Repository 上有两个单独的函数:一个用于fetchFromNetwork,一个用于fetchFromDatabase. MainActivity 应始终从数据库中获取,而 Service 始终触发从网络加载(并通过 Dao 将其直接插入数据库)。

这样,observe当Service在后台线程将数据插入DB时,MainActivity中的函数将接收到最新的数据。

于 2018-07-01T13:26:33.810 回答