• AlarmManager(闹钟服务)


    1.Timer类与AlarmManager类区别:

    对Timer就是定时器,一般写定时任务的时候 肯定离不开他,但是在Android里,他却有个短板,不太适合那些需要长时间在后台运行的 定时任务,因为Android设备有自己的休眠策略,当长时间的无操作,设备会自动让CPU进入 休眠状态,这样就可能导致Timer中的定时任务无法正常运行!而AlarmManager则不存在 这种情况,因为他具有唤醒CPU的功能,可以保证每次需要执行特定任务时CPU都能正常工作, 或者说当CPU处于休眠时注册的闹钟会被保留(可以唤醒CPU),但如果设备被关闭,或者重新 启动的话,闹钟将被清除!(Android手机关机闹钟不响...)

    2.获得AlarmManager实例对象:

    AlarmManager alarmManager = (AlarmManager) getSystemService(ALARM_SERVICE);

    3.相关方法讲解:

    • set(int type,long startTime,PendingIntent pi):一次性闹钟
    • setRepeating(int type,long startTime,long intervalTime,PendingIntent pi): 重复性闹钟,和3有区别,3闹钟间隔时间不固定
    • setInexactRepeating(int type,long startTime,long intervalTime,PendingIntent pi): 重复性闹钟,时间不固定
    • cancel(PendingIntent pi):取消AlarmManager的定时服务
    • getNextAlarmClock():得到下一个闹钟,返回值AlarmManager.AlarmClockInfo
    • setAndAllowWhileIdle(int type, long triggerAtMillis, PendingIntent operation) 和set方法类似,这个闹钟运行在系统处于低电模式时有效
    • setExact(int type, long triggerAtMillis, PendingIntent operation): 在规定的时间精确的执行闹钟,比set方法设置的精度更高
    • setTime(long millis):设置系统墙上的时间
    • setTimeZone(String timeZone):设置系统持续的默认时区
    • setWindow(int type, long windowStartMillis, long windowLengthMillis, PendingIntent operation): 设置一个闹钟在给定的时间窗触发。类似于set,该方法允许应用程序精确地控制操作系统调 整闹钟触发时间的程度。

    关键参数讲解

    • Type(闹钟类型): 有五个可选值: AlarmManager.ELAPSED_REALTIME: 闹钟在手机睡眠状态下不可用,该状态下闹钟使用相对时间(相对于系统启动开始),状态值为3; AlarmManager.ELAPSED_REALTIME_WAKEUP 闹钟在睡眠状态下会唤醒系统并执行提示功能,该状态下闹钟也使用相对时间,状态值为2; AlarmManager.RTC 闹钟在睡眠状态下不可用,该状态下闹钟使用绝对时间,即当前系统时间,状态值为1; AlarmManager.RTC_WAKEUP 表示闹钟在睡眠状态下会唤醒系统并执行提示功能,该状态下闹钟使用绝对时间,状态值为0; AlarmManager.POWER_OFF_WAKEUP 表示闹钟在手机关机状态下也能正常进行提示功能,所以是5个状态中用的最多的状态之一,该状态下闹钟也是用绝对时间,状态值为4;不过本状态好像受SDK版本影响,某些版本并不支持;
    • startTime:闹钟的第一次执行时间,以毫秒为单位,可以自定义时间,不过一般使用当前时间。 需要注意的是,本属性与第一个属性(type)密切相关,如果第一个参数对应的闹钟使用的是相对时间 (ELAPSED_REALTIME和ELAPSED_REALTIME_WAKEUP),那么本属性就得使用相对时间 (相对于系统启动时间来说),比如当前时间就表示为:SystemClock.elapsedRealtime(); 如果第一个参数对应的闹钟使用的是绝对时间(RTC、RTC_WAKEUP、POWER_OFF_WAKEUP), 那么本属性就得使用绝对时间,比如当前时间就表示 为:System.currentTimeMillis()。
    • intervalTime:表示两次闹钟执行的间隔时间,也是以毫秒为单位.
    • PendingIntent:绑定了闹钟的执行动作,比如发送一个广播、给出提示等等。 PendingIntent是Intent的封装类。需要注意的是,如果是通过启动服务来实现闹钟提 示的话,PendingIntent对象的获取就应该采用Pending.getService (Context c,int i,Intent intent,int j)方法;如果是通过广播来实现闹钟 提示的话,PendingIntent对象的获取就应该采用 PendingIntent.getBroadcast (Context c,int i,Intent intent,int j)方法;如果是采用Activity的方式来实 现闹钟提示的话,PendingIntent对象的获取就应该采用 PendingIntent.getActivity(Context c,int i,Intent intent,int j)方法。 如果这三种方法错用了的话,虽然不会报错,但是看不到闹钟提示效果。

    示例:

    package com.loaderman.kotlindemo
    
    import android.app.AlarmManager
    import android.app.PendingIntent
    import android.app.TimePickerDialog
    import android.content.Context
    import android.content.Intent
    import android.os.Bundle
    import android.support.v7.app.AppCompatActivity
    import android.view.View
    import android.widget.Button
    import android.widget.Toast
    import java.util.*
    
    
    class MainActivity : AppCompatActivity() {
    
        var btn_set :Button? = null
    
        override fun onCreate(savedInstanceState: Bundle?) {
            super.onCreate(savedInstanceState)
            setContentView(R.layout.activity_main)
            bindViews()
        }
    
        private fun bindViews() {
            btn_set = findViewById<Button>(R.id.btn_set)
            val btn_cancel = findViewById<View>(R.id.btn_cancel) as Button
            val alarmManager: AlarmManager = getSystemService(Context.ALARM_SERVICE) as AlarmManager
    
            val intent = Intent(this, ClockActivity::class.java)
            val pi = PendingIntent.getActivity(this, 0, intent, 0)
            btn_set?.setOnClickListener(View.OnClickListener {
                val currentTime = Calendar.getInstance()
                TimePickerDialog(this@MainActivity, 0,
                        TimePickerDialog.OnTimeSetListener { view, hourOfDay, minute ->
                            //设置当前时间
                            val c = Calendar.getInstance()
                            c.setTimeInMillis(System.currentTimeMillis())
                            // 根据用户选择的时间来设置Calendar对象
                            c.set(Calendar.HOUR, hourOfDay)
                            c.set(Calendar.MINUTE, minute)
                            // ②设置AlarmManager在Calendar对应的时间启动Activity
                            alarmManager.set(AlarmManager.RTC_WAKEUP, c.getTimeInMillis(), pi)
    
                            // 提示闹钟设置完毕:
                            Toast.makeText(this@MainActivity, "闹钟设置完毕~" + c.getTimeInMillis(),
                                    Toast.LENGTH_SHORT).show()
                        }, currentTime.get(Calendar.HOUR_OF_DAY), currentTime.get(Calendar.MINUTE), false).show()
                btn_cancel.visibility = View.VISIBLE
            })
            btn_cancel.setOnClickListener(View.OnClickListener {
                alarmManager.cancel(pi);
                btn_cancel.setVisibility(View.GONE);
                Toast.makeText(this, "闹钟已取消", Toast.LENGTH_SHORT).show();
            })
    
        }
    }
    package com.loaderman.kotlindemo
    
    import android.content.DialogInterface
    import android.media.MediaPlayer
    import android.os.Bundle
    import android.support.v7.app.AlertDialog
    import android.support.v7.app.AppCompatActivity
    
    
    class ClockActivity : AppCompatActivity() {
    
        override fun onCreate(savedInstanceState: Bundle?) {
            super.onCreate(savedInstanceState)
            setContentView(R.layout.activity_clock)
            val mediaPlayer  = MediaPlayer.create(this, R.raw.tishi)
    
            mediaPlayer.start();
            AlertDialog.Builder(this@ClockActivity).setTitle("闹钟").setMessage("小猪小猪快起床~")
                    .setPositiveButton("关闭闹铃", DialogInterface.OnClickListener { dialog, which ->
                        mediaPlayer.stop()
                        this@ClockActivity.finish()
                    }).show()
    
        }
    }

    activity_main.xml

    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:id="@+id/LinearLayout1"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="vertical">
    
        <Button
            android:id="@+id/btn_set"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="设置闹钟" />
    
        <Button
            android:id="@+id/btn_cancel"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="关闭闹钟"
            android:visibility="gone" />
    
    </LinearLayout>

    核心流程如下:

    • AlarmManager alarmManager = (AlarmManager) getSystemService(ALARM_SERVICE); 获得系统提供的AlarmManager服务的对象
    • Intent设置要启动的组件: Intent intent = new Intent(MainActivity.this, ClockActivity.class);
    • PendingIntent对象设置动作,启动的是Activity还是Service,又或者是广播! PendingIntent pi = PendingIntent.getActivity(MainActivity.this, 0, intent, 0);
    • 调用AlarmManager的set( )方法设置单次闹钟的闹钟类型,启动时间以及PendingIntent对象! alarmManager.set(AlarmManager.RTC_WAKEUP,c.getTimeInMillis(), pi);

    示例可能根据手机厂商和Android系统版本不同,导致无法看到效果,建议在4.4上一些运行

  • 相关阅读:
    shell脚本编程练习
    linux中()、[]、{}、(())、[[]]等各种括号的使用
    Linux Shell 变量自加
    while read line [linux] shell 学习
    踢出某正在访问的用户||永久禁止某IP访问
    linux设置 自定义脚本开机启动
    syntax error: unexpected end of file完美解决方案
    Linux利用nc命令脚本批量检测服务器指定端口是否开放
    41-贪心算法
    38-动态规划
  • 原文地址:https://www.cnblogs.com/loaderman/p/10172230.html
Copyright © 2020-2023  润新知