【问题标题】:JobScheduler for Nougat/Oreo牛轧糖/奥利奥的 JobScheduler
【发布时间】:2018-08-15 14:58:15
【问题描述】:

我遵循了一些关于 android 中的作业调度的在线教程,我尝试这样做,发现下面的代码编译没有错误。但问题是onPostExecute/doInBackground 返回消息中没有吐司。有什么可能需要的吗?就像我只是运行代码并放置了一些祝酒词,并注意到myScheduler.schedule(myjobInfo); 工作正常,后台任务正在执行。

我的 gradle 文件根本没有动过

compileSdkVersion 26
 defaultConfig {
    applicationId "com.example.umarmalik.myapplication"
    minSdkVersion 21
    targetSdkVersion 26
    versionCode 1
    versionName "1.0"
    testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
}

这是我的 JobScheduler 类

  public class MJobScheduler extends JobService {

private MJobExecutor executor;


@Override
public boolean onStartJob(final JobParameters params) {

executor = new MJobExecutor(){
    @Override
    protected void onPostExecute(String s) {
        Toast.makeText(getApplicationContext(),s,Toast.LENGTH_LONG).show();
       jobFinished(params,true);
    }
};
    executor.execute();
    return true;
}

@Override
public boolean onStopJob(JobParameters params) {
    executor.cancel(true);
    return false;
}
}

这是我的 JobExecutor

public class MJobExecutor extends AsyncTask<Void,Void,String> {

@Override
protected String doInBackground(Void... voids) {
    return "Background task running";
}
}

我的主要活动类如下:

public class MainActivity extends AppCompatActivity {


private static final int JOBID = 110;
private JobScheduler myScheduler;
private JobInfo myjobInfo;

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

    ComponentName myComp = new ComponentName(this,MJobScheduler.class);
    JobInfo.Builder myBuilder = new JobInfo.Builder(JOBID,myComp);

    myBuilder.setPeriodic(6000);
    myBuilder.setRequiredNetworkType(JobInfo.NETWORK_TYPE_ANY);
    myBuilder.setPersisted(true);


    myScheduler = (JobScheduler) getSystemService(JOB_SCHEDULER_SERVICE);
    myjobInfo = myBuilder.build();


    myScheduler.schedule(myjobInfo);
}
}

我的清单

<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />

<application
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:roundIcon="@mipmap/ic_launcher_round"
    android:supportsRtl="true"
    android:theme="@style/AppTheme">
    <activity android:name=".MainActivity">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>

    <service
        android:name=".MJobScheduler"
        android:permission="android.permission.BIND_JOB_SERVICE"
        android:exported="true" />

</application>

【问题讨论】:

  • 不要使用AsyncTask。充其量,那是用于 UI 的东西,你需要在主应用程序线程上做一些事情。使用普通的Thread。此外,setPeriodic() 不支持 6 秒周期。除此之外,正如 Viktoriia 所说,使用 Toast 以外的东西进行调试:Log 语句、断点等。
  • “setPeriodic() 不支持 6 秒周期”那么我可以使用的数字是多少?
  • 您可以在JobInfo 上致电getMinPeriodMillis() 以查找您正在运行的设备的最短时间。在许多版本中,它是 15 分钟(例如,请参阅 the Android 8.1 source)。
  • 假设如果我现在开始工作,我什么时候会收到工作完成的通知?是在 15 分钟内还是正好 15 分钟内?

标签: android api android-studio android-jobscheduler jobservice


【解决方案1】:

根据这个答案,定期工作的最短时间是 15 分钟,这就是它不适合你的原因。 https://stackoverflow.com/a/48337856/5604896

在作业完成后更新活动的更好方法是使用广播接收器。如果活动正在运行,则会收到通知。

public class MJobScheduler extends JobService {

    private MJobExecutor executor;


    @Override
    public boolean onStartJob(final JobParameters params) {

        executor = new MJobExecutor(this, params);
        executor.execute();
        return true;
    }

    @Override
    public boolean onStopJob(JobParameters params) {
        executor.cancel(true);
        return false;
    }

    private static class MJobExecutor extends AsyncTask<Void, Void, String> {
        private WeakReference<JobService> jobServiceReference;
        private JobParameters params;

        MJobExecutor(JobService jobService, JobParameters params) {
            jobServiceReference = new WeakReference<>(jobService);
            this.params = params;
        }

        @Override
        protected String doInBackground(Void... voids) {
            return "Background task running";
        }

        @Override
        protected void onPostExecute(String s) {
            super.onPostExecute(s);
            JobService jobService = jobServiceReference.get();
            if (jobService != null) {
                Intent intent = new Intent("JOB_FINISHED");
                intent.putExtra("result", s);
                LocalBroadcastManager.getInstance(jobService).sendBroadcast(intent);
                jobService.jobFinished(params, true);
            }
        }
    }
}

还有你的活动:

public class MainActivity extends AppCompatActivity {


    private static final int JOBID = 110;
    private JobScheduler myScheduler;
    private JobInfo myjobInfo;

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

        ComponentName myComp = new ComponentName(this, MJobScheduler.class);
        JobInfo.Builder myBuilder = new JobInfo.Builder(JOBID, myComp);
        myBuilder.setPeriodic(15 * 60 * 1000);

        myBuilder.setRequiredNetworkType(JobInfo.NETWORK_TYPE_ANY);
        myBuilder.setPersisted(true);

        myScheduler = (JobScheduler) getSystemService(JOB_SCHEDULER_SERVICE);
        myjobInfo = myBuilder.build();


        myScheduler.schedule(myjobInfo);

        LocalBroadcastManager.getInstance(this).registerReceiver(broadcastReceiver, new IntentFilter("JOB_FINISHED"));
    }

    private BroadcastReceiver broadcastReceiver = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {
            Toast.makeText(context, intent.getStringExtra("result"), Toast.LENGTH_LONG).show();
        }
    };

    @Override
    protected void onDestroy() {
        super.onDestroy();
        LocalBroadcastManager.getInstance(this).unregisterReceiver(broadcastReceiver);
    }
}

【讨论】:

  • 谢谢你的效果更好,但是关于后台运行任务的间隔,它是不一致的。例如,出现在后台运行,然后在 5 秒后再次出现,然后在 30 秒后再次出现 1 分钟............有什么解释吗?还有吐司是从哪里来的?
  • @CigarDon 关于间隔 - 它不应该是你描述的方式。您可以尝试删除该应用程序并重新安装它,它会一样吗?它应该每 15 分钟或更短时间触发一次 - 取决于您设置的时间间隔。根据文档“setPeriodic(long intervalMillis) - 指定此作业应以提供的间隔重复,每个周期不超过一次。您无法控制在此间隔内何时执行此作业,只能保证它将在此时间间隔内最多执行一次。"
  • @CigarDon 关于第二个问题,我不确定我是否理解。 “吐司从哪里来”是什么意思?
猜你喜欢
  • 1970-01-01
  • 2020-02-09
  • 1970-01-01
  • 2021-06-23
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2018-08-28
相关资源
最近更新 更多