美文网首页
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