Повтор будильника в Android 8 - PullRequest
0 голосов
/ 28 мая 2018

Не могли бы вы объяснить, почему этот код хорошо работает на Android Lollipop, но я плохо работаю на Android Oreo?Код устанавливает сигнал тревоги, который начинается через 5 секунд и повторяется каждые 4 секунды.Он посылает трансляцию, а MyReceive показывает тост.В Android 8 он начинается в конце, а следующая трансляция отправляется не каждые 4 секунды, а в произвольное время (в конце).Что я могу сделать?

import android.app.AlarmManager;
import android.app.PendingIntent;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.media.AudioManager;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;

import java.util.Calendar;

public class MainActivity extends AppCompatActivity {



    private final String LOG_TEST = "LOG_TEST";
    private Button bStart, bStop;
    private Intent intent;
    private PendingIntent pendingIntent;
    public static final String MY_CUSTOM_ACTION = "mio.broadcast";

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

        final AlarmManager alarm = (AlarmManager) getApplicationContext().getSystemService(Context.ALARM_SERVICE);
        intent = new Intent(MainActivity.this, MyReceiver.class);
        pendingIntent = PendingIntent.getBroadcast(MainActivity.this,0,intent,0);

        bStart=(Button)findViewById(R.id.bStart);
        bStop= (Button)findViewById(R.id.bStop);


        bStart.setOnClickListener(new View.OnClickListener(){

            @Override
            public void onClick(View v) {

                Calendar attivazione = Calendar.getInstance();

                alarm.setRepeating(AlarmManager.RTC_WAKEUP, attivazione.getTimeInMillis()+5000,4000,pendingIntent);
            }
        });

        bStop.setOnClickListener(new View.OnClickListener(){

            @Override
            public void onClick(View v) {
                if(alarm!=null){
                    alarm.cancel(pendingIntent);
                }
            }
        });


    }


}

MyReceiver

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.widget.Toast;

public class MyReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        Toast.makeText(context, "Vibrazione inserita", Toast.LENGTH_LONG).show();
    }
}

Манифест

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.redwi.alerttry">

    <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>

        <receiver
            android:name=".MyReceiver"
            android:enabled="true"
            android:exported="false" />
    </application>

</manifest>

Я надеюсь, что вы можете помочь мне, спасибо!

Ответы [ 2 ]

0 голосов
/ 28 мая 2018

Точность setRepeating не гарантируется быть точной и может непредсказуемо варьироваться от устройства к устройству.Если вам нужно более точное планирование, используйте setExact и перепланируйте будильник самостоятельно.Из документов :

Примечание. Начиная с API 19 (KITKAT), доставка сигналов тревоги является неточной: ОС будет сдвигать сигналы тревоги, чтобы минимизировать пробуждения и использование батареи.Существуют новые API для поддержки приложений, которые требуют строгих гарантий доставки;см. setWindow (int, long, long, PendingIntent) и setExact (int, long, PendingIntent).Приложения, для которых targetSdkVersion более ранний, чем API 19, будут по-прежнему видеть предыдущее поведение, при котором все тревоги доставляются точно по запросу.

0 голосов
/ 28 мая 2018

Не могли бы вы объяснить, почему этот код хорошо работает на Android Lollipop, но я плохо работаю на Android Oreo?

AlarmManager не предназначен для выполнения таких частых задач.Примечания:

  • На Android 4.4+, setRepeating() будет неточным, что означает, что сигнал тревоги может возникнуть в любом месте в течение периода

  • Вкл.Android 5.1+, setRepeating() будет ограничен минимальным периодом в одну минуту

  • В Android 6.0+ ваши тревоги будут зависеть от режима ожидания и, возможно, ожидания приложения, вызываяони запускаются нечасто, когда устройство не находится на зарядном устройстве и не движется

Что я могу сделать?

Если вы хотите показатьToast каждые несколько секунд от переднего плана, используйте postDelayed():

/***
  Copyright (c) 2012 CommonsWare, LLC
  Licensed under the Apache License, Version 2.0 (the "License"); you may not
  use this file except in compliance with the License. You may obtain a copy
  of the License at http://www.apache.org/licenses/LICENSE-2.0. Unless required
  by applicable law or agreed to in writing, software distributed under the
  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS
  OF ANY KIND, either express or implied. See the License for the specific
  language governing permissions and limitations under the License.

  Covered in detail in the book _The Busy Coder's Guide to Android Development_
    https://commonsware.com/Android
 */

package com.commonsware.android.post;

import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.widget.Toast;

public class PostDelayedDemo extends Activity implements Runnable {
  private static final int PERIOD=5000;
  private View root=null;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);
    root=findViewById(android.R.id.content);
  }

  @Override
  public void onStart() {
    super.onStart();

    run();
  }

  @Override
  public void onStop() {
    root.removeCallbacks(this);

    super.onStop();
  }

  @Override
  public void run() {
    Toast.makeText(PostDelayedDemo.this, "Who-hoo!", Toast.LENGTH_SHORT)
         .show();
    root.postDelayed(this, PERIOD);
  }
}

(из этот пример проекта )

...