欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

android service實(shí)現(xiàn)循環(huán)定時(shí)提醒功能

 更新時(shí)間:2018年01月25日 11:20:03   作者:逆天的代碼者  
這篇文章主要為大家詳細(xì)介紹了android service實(shí)現(xiàn)循環(huán)定時(shí)提醒功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下

人每天都要喝8杯水才能保持健康,于是苦逼的程序員總是一遍代碼就忘了時(shí)間,于是我突發(fā)奇想能不能開發(fā)一個(gè)apk能夠?qū)崿F(xiàn)固定的間隔時(shí)間定時(shí)提醒我要喝水了呢?

apk基本功能:

1)能夠設(shè)置間隔時(shí)間 2)在apk應(yīng)用被停止的情況下仍然能定時(shí)提醒 3)能夠播放指定鬧鈴 4)能夠及時(shí)終止提醒

效果圖:

設(shè)置間隔

時(shí)間到后會(huì)跳出全局AlertDialog提示并且開始播放鬧鈴

即使APP被終止了,仍然能夠提示

結(jié)束提示

廢話不多說,直接上代碼:

布局layout:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  xmlns:tools="http://schemas.android.com/tools"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  tools:context="bai.cslg.servicebestpractice.MainActivity"
  android:baselineAligned="false"
  android:orientation="vertical">


  <LinearLayout
    android:paddingTop="20dp"
    android:layout_width="match_parent"
    android:layout_height="70dp">
    <TextView
      android:layout_width="0dp"
      android:layout_height="wrap_content"
      android:layout_weight="5"
      android:padding="10dp"
      android:gravity="center_vertical"
      android:text="請(qǐng)?jiān)O(shè)置提示時(shí)間間隔:"
      android:textSize="20sp"/>
    <EditText
      android:id="@+id/time"
      android:layout_width="0dp"
      android:layout_height="wrap_content"
      android:layout_weight="1"/>
    <TextView
      android:layout_width="0dp"
      android:layout_height="wrap_content"
      android:layout_weight="2"
      android:gravity="center_vertical"
      android:text="分"
      android:textSize="20sp"/>
  </LinearLayout>

  <LinearLayout
    android:layout_width="match_parent"
    android:layout_height="wrap_content">
  <Button
    android:id="@+id/start_serice"
    android:layout_width="0dp"
    android:layout_height="wrap_content"
    android:layout_weight="1"
    android:text="開啟"/>
    <Button
      android:id="@+id/stop_serice"
      android:layout_width="0dp"
      android:layout_height="wrap_content"
      android:layout_weight="1"
      android:text="結(jié)束"/>
  </LinearLayout>
</LinearLayout>

MainActivity代碼:

/*
*因?yàn)橐?wù)常駐后臺(tái),就不需要BindService,直接StartService即可
*/
package bai.cslg.servicebestpractice;

import android.content.Context;
import android.content.Intent;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity implements View.OnClickListener{

  private Context mContext = MainActivity.this;

  private Button startService;
  private Button stopService;
  private EditText time;

  public static int TIME; //記錄時(shí)間間隔

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    startService = (Button) findViewById(R.id.start_serice);
    stopService = (Button) findViewById(R.id.stop_serice);
    time = (EditText) findViewById(R.id.time);

    startService.setOnClickListener(this);
    stopService.setOnClickListener(this);

  }

  @Override
  public void onClick(View view) {
    switch (view.getId()){
      case R.id.start_serice:
        Intent startIntent = new Intent(this,LongRunningService.class);
        TIME = Integer.parseInt(time.getText().toString().trim());
        //通過Intent將時(shí)間間隔傳遞給Service
        startIntent.putExtra("Time",TIME);
        Toast.makeText(MainActivity.this,"開始提醒",Toast.LENGTH_SHORT).show();
        startService(startIntent);
        break;
      case R.id.stop_serice:
        Intent stopIntent = new Intent(this,LongRunningService.class);
        Toast.makeText(MainActivity.this,"結(jié)束提醒",Toast.LENGTH_SHORT).show();
        stopService(stopIntent);
        break;
    }
  }
}

Service代碼:

package bai.cslg.servicebestpractice;

import android.app.AlarmManager;
import android.app.AlertDialog;
import android.app.PendingIntent;
import android.app.Service;
import android.content.DialogInterface;
import android.content.Intent;
import android.media.MediaPlayer;
import android.os.Handler;
import android.os.IBinder;
import android.os.Message;
import android.os.SystemClock;
import android.support.annotation.Nullable;
import android.util.Log;
import android.view.WindowManager;



import java.io.File;
import java.io.IOException;
import java.util.Date;

/**
 * Created by baiqihui on 2016/9/21.
 */
public class LongRunningService extends Service {

  public int anHour; //記錄間隔時(shí)間

  public int number = 0; //記錄alertdialog出現(xiàn)次數(shù)

  private MediaPlayer mediaPlayer = new MediaPlayer();

  AlarmManager manager;
  PendingIntent pi;


  private Handler mHandler = new Handler(){
    @Override
    public void handleMessage(Message msg) {
      super.handleMessage(msg);
      switch (msg.what){
        case 1:
          if (!mediaPlayer.isPlaying()){
            mediaPlayer.start();
          }
          AlertDialog.Builder builder = new AlertDialog.Builder(LongRunningService.this);
          builder.setTitle("提醒");
          builder.setMessage("該補(bǔ)水啦" + (number-1));
          builder.setCancelable(false);
          builder.setPositiveButton("OK", new DialogInterface.OnClickListener() {
            @Override
            public void onClick(DialogInterface dialogInterface, int i) {
              mediaPlayer.reset();
              initMediaPlayer();
            }
          });
          final AlertDialog dialog = builder.create();
          dialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
          dialog.show();
      }
    }
  };

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

  @Override
  public void onCreate() {
    super.onCreate();
    initMediaPlayer();

  }

  private void initMediaPlayer() {
    File file = new File("/storage/emulated/0/naoling","music.mp3");
    try {
      mediaPlayer.setDataSource(file.getPath());
      mediaPlayer.prepare();
    } catch (IOException e) {
      e.printStackTrace();
    }
  }

  @Override
  public int onStartCommand(Intent intent, int flags, int startId) {
    if (number!=0) {
      new Thread(new Runnable() {
        @Override
        public void run() {
          Log.e("bai", "executed at " + new Date().toString());
          mHandler.sendEmptyMessage(1);
        }
      }).start();
    }
    manager = (AlarmManager) getSystemService(ALARM_SERVICE);
    int time = intent.getIntExtra("Time",2);
    anHour = time*60*1000;
    Log.e("bai","Time:"+time+"anhour:"+anHour);
    long triggerAtTime = SystemClock.elapsedRealtime()+(anHour);
    Intent i = new Intent(this,AlarmReceiver.class);
    pi = PendingIntent.getBroadcast(this,0,i,0);
    manager.set(AlarmManager.ELAPSED_REALTIME_WAKEUP,triggerAtTime,pi);
    number++;
    return super.onStartCommand(intent, flags, startId);
  }

  @Override
  public void onDestroy() {
    super.onDestroy();
    mediaPlayer.release();
    manager.cancel(pi);
  }
}

AlarmReceiver代碼:

package bai.cslg.servicebestpractice;

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

/**
 * Created by baiqihui on 2016/9/21.
 */
public class AlarmReceiver extends BroadcastReceiver {
  @Override
  public void onReceive(Context context, Intent intent) {
      Intent i = new Intent(context, LongRunningService.class);
      context.startService(i);
  }
}

1)在apk應(yīng)用被停止的情況下仍然能定時(shí)提醒,這里采用startService即可實(shí)現(xiàn),使service常駐內(nèi)存,即使Activity被殺死,依舊可以執(zhí)行。

2)間隔時(shí)間提醒。這里采用的是Android的Alarm機(jī)制。

 Android中的定時(shí)任務(wù)一般有兩種實(shí)現(xiàn)方式,一種是使用Java API里提供的Timer類,一種是使用Android的Alarm機(jī)制。這兩種情況在多數(shù)情況下都能實(shí)現(xiàn)類似的效果,但Timer類有一個(gè)明顯的短板,它并不太適用于那些需要長(zhǎng)期在后臺(tái)運(yùn)行的定時(shí)任務(wù)。我們都知道,為了能讓電池更耐用,每種手機(jī)都會(huì)有自己的休眠策略,Android手機(jī)就會(huì)在長(zhǎng)時(shí)間不操作的情況下自動(dòng)讓CPU進(jìn)入到睡眠狀態(tài),這就有可能導(dǎo)致Timer中的定時(shí)任務(wù)無法正常運(yùn)行。而Alarm機(jī)制則不存在這種情況,它具有喚醒CPU的功能,即可以保證每次需要執(zhí)行定時(shí)任務(wù)的時(shí)候CPU都能正常工作。需要注意,這里喚醒CPU和喚醒屏幕完全不是一個(gè)概念。

從Service代碼中可以看出,onCreate()中完成對(duì)mediaPlayer的初始化(因?yàn)閙ediaPlayer只需要初始化一次),在onStartCommand()中開啟一個(gè)新的線程,線程中通過handler發(fā)送一條空的消息,并且在handler的handleMessage()方法中完成AlertDialog的創(chuàng)建以及播放鬧鈴,要注意這里創(chuàng)建的是一個(gè)全局的AlertDialog。因?yàn)榈谝淮伍_啟任務(wù)的時(shí)候不需要新建一個(gè)AlertDialog(用戶第一次開啟任務(wù)的時(shí)候是設(shè)置好時(shí)間并且點(diǎn)擊了“開啟”,這個(gè)時(shí)候不需要?jiǎng)?chuàng)建Dialog)。

在onStartCommand()還執(zhí)行了AlarmManager的初始化以及時(shí)間的設(shè)定,因?yàn)锳larmManager中第三個(gè)參數(shù)PendingIntent能夠執(zhí)行一個(gè)廣播,所以還需要寫一個(gè)廣播接收者。

AlarmManager的取消:manager.cancel(PendingIntent pi);取消對(duì)應(yīng)PendingIntent即可。

AlarmReceiver:這就很簡(jiǎn)單了,接收到廣播之后開啟再開啟服務(wù)即可。這就詳單與是一個(gè)死循環(huán),服務(wù)開啟后會(huì)定時(shí)發(fā)送廣播,廣播接收到之后又會(huì)開啟服務(wù)。

因?yàn)闀r(shí)間有限,所以代碼肯定有很多不完善之處,希望多多指教。

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Android MenuItem 自定義長(zhǎng)按事件的實(shí)現(xiàn)

    Android MenuItem 自定義長(zhǎng)按事件的實(shí)現(xiàn)

    這篇文章主要介紹了Android MenuItem 自定義長(zhǎng)按事件的實(shí)現(xiàn),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-08-08
  • 新版Android studio導(dǎo)入微信支付和支付寶官方Demo問題解決大全

    新版Android studio導(dǎo)入微信支付和支付寶官方Demo問題解決大全

    這篇文章主要為大家詳細(xì)介紹了新版Android studio導(dǎo)入微信支付和支付寶官方Demo問題的解決大全,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2016-07-07
  • OKhttp攔截器實(shí)現(xiàn)實(shí)踐環(huán)節(jié)源碼解析

    OKhttp攔截器實(shí)現(xiàn)實(shí)踐環(huán)節(jié)源碼解析

    這篇文章主要為大家介紹了OKhttp攔截器實(shí)現(xiàn)實(shí)踐環(huán)節(jié)源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-01-01
  • Android 開源項(xiàng)目側(cè)邊欄菜單(SlidingMenu)使用詳解

    Android 開源項(xiàng)目側(cè)邊欄菜單(SlidingMenu)使用詳解

    SlidingMenu的是一種比較新的設(shè)置界面或配置界面效果,在主界面左滑或者右滑出現(xiàn)設(shè)置界面,能方便的進(jìn)行各種操作.目前有大量的應(yīng)用都在使用這一效果。如Evernote、Google+、Foursquare等,國(guó)內(nèi)的豌豆夾,人人,360手機(jī)助手等都使用SlidingMenu的界面方案。
    2016-05-05
  • 準(zhǔn)確測(cè)量 Android 應(yīng)用中 Activity 和 Fragment 的啟動(dòng)時(shí)間的詳細(xì)過程

    準(zhǔn)確測(cè)量 Android 應(yīng)用中 Activity 和 Fragmen

    在 Android 應(yīng)用開發(fā)中,了解每個(gè) Activity 和 Fragment 的啟動(dòng)時(shí)間對(duì)于性能優(yōu)化至關(guān)重要,本文將介紹幾種方法來準(zhǔn)確測(cè)量 Activity 和 Fragment 的啟動(dòng)時(shí)間,并提供實(shí)際操作步驟,以幫助提升應(yīng)用的響應(yīng)速度和用戶體驗(yàn),需要的朋友可以參考下
    2024-07-07
  • Flutter開發(fā)之Widget自定義總結(jié)

    Flutter開發(fā)之Widget自定義總結(jié)

    這篇文章主要給大家介紹了關(guān)于Flutter開發(fā)中Widget自定義的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Flutter具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-04-04
  • Android如何跳轉(zhuǎn)到應(yīng)用商店的APP詳情頁面

    Android如何跳轉(zhuǎn)到應(yīng)用商店的APP詳情頁面

    最近做項(xiàng)目遇到這樣的需求,要求從App內(nèi)部點(diǎn)擊按鈕或鏈接,跳轉(zhuǎn)到應(yīng)用商店的某個(gè)APP的詳情頁面,怎么實(shí)現(xiàn)此功能呢?下面小編給大家分享Android如何跳轉(zhuǎn)到應(yīng)用商店的APP詳情頁面,需要的朋友參考下
    2017-01-01
  • Android 服務(wù)端將位置信息發(fā)送給客戶端的實(shí)現(xiàn)

    Android 服務(wù)端將位置信息發(fā)送給客戶端的實(shí)現(xiàn)

    這篇文章主要介紹了Android 服務(wù)端將位置信息發(fā)送給客戶端的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-01-01
  • 關(guān)于Android中WebView遠(yuǎn)程代碼執(zhí)行漏洞淺析

    關(guān)于Android中WebView遠(yuǎn)程代碼執(zhí)行漏洞淺析

    這篇文章主要給大家介紹了關(guān)于Android中WebView遠(yuǎn)程代碼執(zhí)行漏洞的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2018-05-05
  • 去掉RecycleView或者ListView上下滑動(dòng)陰影的方法

    去掉RecycleView或者ListView上下滑動(dòng)陰影的方法

    下面小編就為大家分享一篇去掉RecycleView或者ListView上下滑動(dòng)陰影的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2018-01-01

最新評(píng)論