Notification除了用于后台服务通知,还常用在下面情况:

(1)保持服务存在。当系统内存不足时,系统会认为某后台服务占用内存时间太长而中止该服务,以释放内存。对于某些服务,例如播放音乐,如果系统对该服务进行资源释放,用户体验就成了音乐突然没有声音。对这类服务,我们希望享有更高的优先级别,不会被系统干掉。

(2)用户随时与服务进行互动。例如播放音乐的服务,用户可随时暂停音乐播放,或选择其他曲目,甚至中止播放音乐服务。

要实现上述两点,方法是在Service中宣称自己是foreground,并维持一个通知来向用户表明foreground状态,用户可以通过下拉通知并点击通知,进入该服务界面,实施互动操作。

我们将在Android学习笔记(五二):服务Service(中)- 继承Service类的模拟音乐播放例子上进行扩展。

Android学习笔记(五五):通知Notification(下)_播放音乐

FakePlayer同原例子的客户端代码,略过不表。Service的代码如下

public class FakePlayerService extends Service{ 
    public static final String EXTRA_PLAYLIST="EXTRA_PLAYLIST";
     public static final String EXTRA_SHUFFLE="EXTRA_SHUFFLE"; 
     private boolean isPlay = false; 
     private static final int NOTIFY_FAKEPLAYER_ID=1339;  
     
    public IBinder onBind(Intent arg0) {  
         return null; 
     } 

     public void onDestroy() {  
         stop(); 
     } 

     public int onStartCommand(Intent intent, int flags, int startId) {  
        String playlist = intent.getStringExtra(EXTRA_PLAYLIST);
         Boolean isShuffle = intent.getBooleanExtra(EXTRA_SHUFFLE, false); 
         play(playlist,isShuffle); 
        return START_NOT_STICKY; 
     } 
     
     private void play(String playlist, Boolean isShuffle){  
        if(!isPlay){  
             isPlay = true; 
             //和上一笔记中创建通知的步骤一样,只是不需要通过通知管理器进行触发,而是用startForeground(ID,notify)来处理
            //步骤1:和上一笔记一样,通过Notification.Builder( )来创建通知
            //FakePlayer就是两个大button的activity,也即服务的界面,见最左图
            Intent i = new Intent(this,FakePlayer.class);
             //注意Intent的flag设置:FLAG_ACTIVITY_CLEAR_TOP: 如果activity已在当前任务中运行,在它前端的activity都会被关闭,它就成了最前端的activity。FLAG_ACTIVITY_SINGLE_TOP: 如果activity已经在最前端运行,则不需要再加载。设置这两个flag,就是让一个且唯一的一个activity(服务界面)运行在最前端。
             i.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_SINGLE_TOP);            PendingIntent pi = PendingIntent.getActivity(this, 0, i, 0);           
             Notification myNotify = new Notification.Builder(this) 
                                     .setSmallIcon(R.drawable.shield) 
                                     .setTicker("Fake Player: " + playlist) 
                                     .setContentTitle("Test") 
                                     .setContentText("Now Playing: \"Ummmm, Nothing\"")
                                     .setContentIntent(pi) 
                                     .build();             
             //设置notification的flag,表明在点击通知后,通知并不会消失,也在最右图上仍在通知栏显示图标。这是确保在activity中退出后,状态栏仍有图标可提下拉、点击,再次进入activity。
             myNotify.flags |= Notification.FLAG_NO_CLEAR;
             
            // 步骤 2:startForeground( int, Notification)将服务设置为foreground状态,使系统知道该服务是用户关注,低内存情况下不会killed,并提供通知向用户表明处于foreground状态。
             startForeground(NOTIFY_FAKEPLAYER_ID,myNotify);
         } 
     } 
     
     private void stop(){ 
        if(isPlay){ 
             isPlay = false; 
            //将服务从forefround状态中移走,使得系统可以在低内存的情况下清除它。 (true);  
        } 
     } 

 }