搜索

首页  >  问答  >  正文

android - 安卓AlarmManager.setRepeating()真机运行的结果和设定的循环时间间隔不一致

用了一下安卓的AlarmManager,写了一个很简单的例子程序,可是结果和预想的不一样,代码如下
主页面:

package com.example.athand.alarmmanagertest;

import android.app.AlarmManager;
import android.app.PendingIntent;
import android.content.Intent;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;

public class MainActivity extends AppCompatActivity {
    private Button setBtn,cancelBtn;
    private static final String BC_ACTION="com.example.athand.alarmmanagertest.BC_ACTION";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        setBtn=(Button)findViewById(R.id.setBtn);
        cancelBtn=(Button)findViewById(R.id.cancelBtn);
        final AlarmManager am=(AlarmManager)getSystemService(ALARM_SERVICE);
        Intent intent=new Intent();
        intent.setAction(BC_ACTION);
        intent.putExtra("msg", "你该去开会了");
        final PendingIntent pi=PendingIntent.getBroadcast(MainActivity.this,0,intent,0);
        final long time=System.currentTimeMillis();
        setBtn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                am.setRepeating(AlarmManager.RTC_WAKEUP,time,3000,pi);
                Log.d("lokk","click here");
                //am.setExact();
            }
        });
        cancelBtn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                am.cancel(pi);
            }
        });
    }
}

一个简单的Receiver页面:

package com.example.athand.alarmmanagertest;

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

/**
 * Created by Administrator on 2015/10/9.
 */
public class MyReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        String msg=intent.getStringExtra("msg");
        Toast.makeText(context,msg,Toast.LENGTH_LONG).show();
    }
}

按道理来说,在主页面按了按钮,3秒以后就应该重复的执行闹钟,可是在小米2真机执行的时候,toast的消息大概10多秒以后才出现,然后就不重复了,不知道哪里出问题,还是我对于setRepeating的理解有误。
ps,小米2的版本是16的,setRepeating不准的问题应该是19才出现的

ringa_leeringa_lee2772 天前613

全部回复(1)我来回复

  • 高洛峰

    高洛峰2017-04-17 14:39:26

    你的代码是可以实现3秒定时的任务的。
    (1) 只执行一次,是不是因为你误点了cancel按钮,取消了定时器。
    (2) 10秒以后才出现,是不是下面的原因(没有小米无法实测),
    看这个问题MiUI AlarmManager 延迟

    自己用的小米手机, 随手拿来测试, 发现每次闹钟都会有5分钟以内的随机的延迟

    最后一个建议:
    在调用setRepeating安排定时器之前,最好检查定时器是否已经在运行,像这样:

    if (isServiceAlarmOn(this)) {
        alarmManager.setRepeating(AlarmManager.RTC, 
                System.currentTimeMillis(), POLL_INTERVAL, pi);
    } else {
        // 取消Alarm,同时取消PendingIntent.
        alarmManager.cancel(pi);
        pi.cancel();
    }
          
    // 由于在取消Alarm的同时也取消了pi,并且一个PendingIntent只能登记给一个Alarm,
    // 所以可通过检查pi是否存在,来确认Alarm是否激活。
    public static boolean isServiceAlarmOn(Context context) {
        Intent i = new Intent(context, PollService.class);
        // FLAG_NO_CREATE表示如果描述的pi不存在,则返回null,而不是创建它。
        PendingIntent pi = PendingIntent.getService(
                context, 0, i, PendingIntent.FLAG_NO_CREATE);
        return pi != null;
    }

    回复
    0
  • 取消回复