即使屏幕被锁定,我如何定期在后台运行服务/线程?

时间:2016-02-25 13:17:54

标签: android multithreading service wakelock

我正在开发一款可以每半小时检查一次网络数据的应用,我需要确保只要电源打开就能继续运行。 现在,我的应用程序的结构是这样的:

  1. main_activity:
    onCreate()
  2. 中的AlarmManager
  3. alarm_receiver:
    start_service 获取服务的partial_wl
  4. 服务:
    使用StrictMode获取网络数据 如果预期数据,则弹出activity_2
  5. activity_2:
    振动 按钮退出(activity_2.this.finish())
  6. 但是在测试中我发现服务将在前30分钟后停止(被杀)。另外,如果我在服务中启动一个线程而不是使用StrictMode,它将在锁定屏幕后的5分钟内被终止。

    希望有人可以为此提出建议。这真的令人不安。 非常感谢。

2 个答案:

答案 0 :(得分:0)

普通服务并不会影响活动的发生。如果你想要它开始定期检查我的服务: https://bitbucket.org/kvrus/ocs-android/raw/036de7f0d3579b2a193bcb82309f7f82819508e6/app/src/main/java/koss/ru/oneclickrate/network/EcbEuropeService.java

/ **  *定期从网络加载汇率  *返回广播消息中的结果。  *由koss于19.02.16创建。  * * / 公共类EcbEuropeService扩展了服务{

public static final String ECB_URL = "http://www.ecb.europa.eu/stats/eurofxref/eurofxref-daily.xml";
public static final int UPDATE_PERIOD = 30000;
public static final int UPDATE_TICK = 1000;

public static final String NOTIFICATION = "koss.ru.oneclickrate.receiver";
public static final String EXTRA_CURRENCIES_MAP = "extra_currencies_map";

@Override
public int onStartCommand(Intent intent, int flags, int startId) {
    getUrlData();
    return Service.START_NOT_STICKY;
}

@Override
public IBinder onBind(Intent intent) {
    return null;
}

public Cubes getUrlData() {
    (new AsyncTask<Object, Object, Cubes>() {
        Map<CurrencyType, BigDecimal> result = new EnumMap<CurrencyType, BigDecimal>(CurrencyType.class);

        @Override
        protected Cubes doInBackground(Object... params) {
            Cubes cubes = new Cubes();
            InputStream is = null;
            HttpURLConnection urlConnection = null;
            try {
                URL url = new URL(ECB_URL);
                urlConnection = (HttpURLConnection) url.openConnection();
                is = urlConnection.getInputStream();
                cubes = EcbEuropeResponseParser.parse(is);
            } catch (Exception e) {
                e.printStackTrace();
            } finally {
                if(urlConnection!=null) IOUtils.close(urlConnection);
                if(is!=null) IOUtils.closeQuietly(is);
                return cubes;
            }
        }

        @Override
        protected void onPostExecute(Cubes map) {
            super.onPostExecute(map);
            sendBroadcastMessage(map);
            startTimer();
        }
    }).execute();
    return null;
}

/**
 * Restarts timer
 * */
public void startTimer() {
    cdt.cancel();
    cdt.start();
}

CountDownTimer cdt = new CountDownTimer(UPDATE_PERIOD, UPDATE_TICK) {
    @Override
    public void onTick(long millisUntilFinished) {

    }

    public void onFinish() {
        getUrlData();
    }
};

private void sendBroadcastMessage(Cubes currenciesMap) {
    Intent intent = new Intent(NOTIFICATION);
    intent.putExtra(EXTRA_CURRENCIES_MAP, currenciesMap);
    sendBroadcast(intent);
}

答案 1 :(得分:0)

我改变了一些事情,现在效果很好。

1.由于我的手机是4.4.2(api = 19),alarmmanager.setrepeating是不准确的。所以我转向使用.setExact(.set()的新方法)并在服务中的AsyncTask(网络)结束时重新安排警报。

2.使wakelock实例全局化,在AlarmReceiver中获取它并在AsyncTask结束时释放。我曾经把.release()放在onDestroy()中,它在任务完成之前释放锁。

3.我的手机中有关于受保护后台应用程序的设置,但我没有将其打开。这可以允许系统终止应用程序并禁用警报管理器。