美文网首页
Notification 配置记录与防止APP被杀总结

Notification 配置记录与防止APP被杀总结

作者: 一个冬季 | 来源:发表于2021-02-05 13:49 被阅读0次
参考文章

仿微信通知栏
Android基础知识(二十):Notification、提醒式通知(横幅)踩坑与通知界面设置跳转
PendingIntent.getActivity的使用

Android平台App进程优先级
Android 程序后台运行和锁屏运行

重要记录点

1、setSmallIcon、setLargeIcon 必须设置。设置小图标的时候,小图标是需要透明背景,实体区域用白色的就可以了,大小64 * 64吧,这个可以修改
2、华为目前是不可以通过代码设置横幅的,想要显示横幅出来,需要自己手动的去通知栏那打开。微信,QQ有钱没办法,天生就可以默认横幅,在notification设置震动也无效,估计所有的开发商都有修改过notification吧
3、可以在线程创建notification
4、开启横幅方式(修改过底层的除开),将通知等级设置最高

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {//android 8.0
      val channel = NotificationChannel(ConversionChannelId, ConversionChannelName, IMPORTANCE_HIGH)
   } else {
      notificationBuilder.setPriority(NotificationCompat.PRIORITY_HIGH)
  }
importance、priority都必须设置到高值

5、设置通知栏不消失的方式

        var notificationBuilder: NotificationCompat.Builder? = null
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {//android 8.0
            notificationBuilder = NotificationCompat.Builder(CommonApplication.getContext(), SingleAppChannelId)
        } else {
            notificationBuilder = NotificationCompat.Builder(CommonApplication.getContext())
        }
        notificationBuilder.let {
            val notification = notificationBuilder.build();
            notification.flags = notification.flags or Notification.FLAG_ONGOING_EVENT//不消失的通知栏
        }

6、如果我不开启前台service,只是将notification设置为不消失的通知栏,按下home键后,主oom_adj的值为11,子进程为15。主与子易被杀死

只开启notification_不消失.jpg

7、主进程开启前台Service,按下home键后,oom_adj的值为3,如果此时有子进程,子进程的oom_adj的值为15。子易被杀死


oom_adj_比较.jpg

8、目前学习到尽量不被杀的办法

1、主进程开启前台service
2、service里面监听开屏、息屏广播。不管开屏、息屏都循环播放无声音乐。目前oppo是无法监听到开屏息屏的通知的,所以需要单独处理,如果是oppo,vivo,直接播放无声音乐好了
3、定时的调用startForeground刷新notification,当然要保证notification的id都相同
4、开启service同时获取PowerManager,并且调用acquire()方法,避免息屏的时候CPU直接睡眠,不执行代码

当然了,你退出APP之后,以上系列的所有操作,都需要进行 一 一 关闭
知识点记录

1、PendingIntent.getActivity知识点

PendingIntent.getActivity(Context context, int requestCode,  intent, int flags)
第一个参数连接上下文的context
第二个参数是对PendingIntent的描述,请求值不同Intent就不同
第三个参数是一个Intent对象,包含跳转目标
第四个参数有4种状态

FLAG_CANCEL_CURRENT:如果当前系统中已经存在一个相同的PendingIntent对象,那么就将先将已有的PendingIntent取消,然后重新生成一个PendingIntent对象。

FLAG_NO_CREATE:如果当前系统中不存在相同的PendingIntent对象,系统将不会创建该PendingIntent对象而是直接返回null。

FLAG_ONE_SHOT:该PendingIntent只作用一次。在该PendingIntent对象通过send()方法触发过后,PendingIntent将自动调用cancel()进行销毁,那么如果你再调用send()方法的话,系统将会返回一个SendIntentException。

FLAG_UPDATE_CURRENT:如果系统中有一个和你描述的PendingIntent对等的PendingInent,那么系统将使用该PendingIntent对象,但是会使用新的Intent来更新之前PendingIntent中的Intent对象数据,例如更新Intent中的Extras。

notification详细代码
 /**
     * @date 创建时间: 2021/2/1
     * @auther gaoxiaoxiong
     * @description 显示信息,这个需要在线程创建,因为头像的获取是线程
     **/
    fun showChatNotification(context: Context) {
        // 创建一个NotificationManager的引用
        val mNotificationManager = context.getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager;
        var notificationBuilder: NotificationCompat.Builder? = null
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {//android 8.0
            notificationBuilder = NotificationCompat.Builder(CommonApplication.getContext(), ConversionChannelId)
            val channel = NotificationChannel(ConversionChannelId, ConversionChannelName, IMPORTANCE_HIGH)
            channel.description = ConversionChannelName;
            mNotificationManager.createNotificationChannel(channel);
        } else {
            notificationBuilder = NotificationCompat.Builder(CommonApplication.getContext())
            notificationBuilder.setPriority(NotificationCompat.PRIORITY_HIGH)
        }

        val intent = Intent(context,MainActivity::class.java);
        intent.putExtra(MainActivity.ID,10);

        notificationBuilder.let {
            it.setWhen(System.currentTimeMillis());
            it.setSmallIcon(R.drawable.icon_sml_notification)//设置小图标,设置后在状态栏部分显示,注意,小图标是需要透明背景的,实体会话部分用白色的就可以了。大小64 * 64吧,这个可以修改
            var userHeaderBitmap = GetImageInputStream(daoChatRecordModel.visitorAvatar);//获取网络图片
            if (userHeaderBitmap == null) {
                userHeaderBitmap = BitmapFactory.decodeResource(CommonApplication.getContext().getResources(), R.mipmap.icon_logo)
            }
            it.setLargeIcon(userHeaderBitmap);//设置大图标
            it.setContentTitle(StringUtils.isEmptyValue(daoChatRecordModel.visitorVame));//设置标题
            it.setContentText(ResourcesUtils.getResourcesString(context, R.string.conversation_shopmessage))//设置文本信息
            //设置当点击通知之后,将要跳转的Activity
            it.setContentIntent(PendingIntent.getActivity(context, 15, intent, PendingIntent.FLAG_UPDATE_CURRENT))
            //弹出通知,悬浮在通知栏
            it.setFullScreenIntent(PendingIntent.getActivity(context, 15, intent, PendingIntent.FLAG_UPDATE_CURRENT), true)
            //设置通知在第一次到达时在状态栏中显示的文本
            it.setTicker(StringUtils.isEmptyValue(daoChatRecordModel.content))
            it.setAutoCancel(true)//可以取消

            it.setVisibility(NotificationCompat.VISIBILITY_PUBLIC)//显示通知的全部内容
            it.setPriority(NotificationManager.IMPORTANCE_MAX)// 当发出此类型的通知时,通知会以悬挂的方法显示在屏幕上
            it.setSound(defaultNoticeVoice);//声音
            it.setVibrate(longArrayOf(0, 500, 1000, 1500));//震动

            val notification = notificationBuilder.build();
            it.setPublicVersion(notification);//安全锁屏下的通知

            mNotificationManager.notify(11, notification)

        }
    }



 /**
     * 获取网络图片
     *
     * @param imageurl 图片网络地址
     * @return Bitmap 返回位图
     */
    private fun GetImageInputStream(imageurl: String): Bitmap? {
        if (StringUtils.isEmpty(imageurl)) {
            return null;
        }
        val url: URL
        var connection: HttpURLConnection? = null
        var bitmap: Bitmap? = null
        try {
            url = URL(imageurl)
            connection = url.openConnection() as HttpURLConnection
            connection.setConnectTimeout(2 * 1000) //超时设置
            connection.setDoInput(true)
            connection.setUseCaches(false) //设置不使用缓存
            val inputStream: InputStream = connection.getInputStream()
            bitmap = BitmapFactory.decodeStream(inputStream)
            inputStream.close()
        } catch (e: Exception) {
            e.printStackTrace()
        }
        return bitmap
    }
 /**
     * @date 创建时间:2021/2/3 0003
     * @auther gaoxiaoxiong
     * @Descriptiion 获取默认的提示音
     **/
    public Uri getDefaultNoticeVoice() {
        Uri notification = RingtoneManager.getDefaultUri(RingtoneManager.TYPE_NOTIFICATION);
        if (notification == null) {//获取本地的
            return Uri.parse("android.resource://" + CommonApplication.getContext().getPackageName() + "/" + R.raw.default_notice_voice);
        } else {
            return notification;
        }
    }

相关文章

网友评论

      本文标题:Notification 配置记录与防止APP被杀总结

      本文链接:https://www.haomeiwen.com/subject/feiwtltx.html