记录一次session安装应用recyclerview更新数据的bug

news2025/6/3 10:07:09

首先抛出异常日志,在

先说结论:因为session安装监听是在点击事件里面,所以会保留旧的对象数据

直接上代码,原有的逻辑是点击时执行session安装,并注册监听回调

  private fun installApk(position: Int) {
        val packageInstaller = packageManager.packageInstaller
        val params =
            PackageInstaller.SessionParams(PackageInstaller.SessionParams.MODE_FULL_INSTALL)
        params.setSize(File(itemList[position].path).length())

        val sessionId = packageInstaller.createSession(params)
        val session = packageInstaller.openSession(sessionId)

        // 注册安装进度监听器
        val sessionCallback = object : PackageInstaller.SessionCallback() {
            override fun onCreated(sessionId: Int) {
                // 安装会话创建成功
                Log.i("InstallProgress", "Session created: $sessionId")
            }

            override fun onProgressChanged(sessionId: Int, progress: Float) {
                // 安装进度更新
                itemList[mPosition].installed = 1
                itemList[mPosition].progress = progress * 100
                adapter.notifyItemChanged(mPosition)
                Log.d(
                    "install",
                    "onProgressChanged: $mPosition ${itemList[mPosition].installed} $progress"
                )
            }

            override fun onFinished(sessionId: Int, success: Boolean) {
                if (success) {
                    // 安装成功
                    itemList[mPosition].installed = 2
                } else {
                    // 安装失败
                    itemList[mPosition].installed = 3
                }
                adapter.notifyItemChanged(mPosition)
                Log.d(
                    "install",
                    "onProgressChanged: $mPosition ${itemList[mPosition].installed} $success"
                )
            }

            override fun onActiveChanged(sessionId: Int, active: Boolean) {
                // 安装会话激活状态发生变化
            }

            override fun onBadgingChanged(sessionId: Int) {
                // 应用图标发生变化
            }
        }
        // 注册监听器
        packageManager.packageInstaller.registerSessionCallback(sessionCallback)

        try {
            val inputStream = FileInputStream(itemList[position].path)
            val outputStream = session.openWrite("base.apk", 0, -1) // -1 表示自动计算大小
            inputStream.copyTo(outputStream)
            session.fsync(outputStream)
            inputStream.close()
            outputStream.close()

            val intent = Intent(context, PackageInstallReceiver::class.java).apply {
                action = itemList[position].pkg
                putExtra(PackageInstaller.EXTRA_PACKAGE_NAME, itemList[position].pkg)
            }

            val pendingIntent = PendingIntent.getBroadcast(
                context,
                0,
                intent,
                PendingIntent.FLAG_UPDATE_CURRENT
            )
            session.commit(pendingIntent.intentSender)
        } catch (e: IOException) {
            session.abandon()
            e.printStackTrace()
        }
    }

修改后,将监听注册到onCreate中,并

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        // 注册监听器
        packageManager.packageInstaller.registerSessionCallback(sessionCallback)
    }

    // 注册安装进度监听器
    private val sessionCallback = object : PackageInstaller.SessionCallback() {
        override fun onCreated(sessionId: Int) {
            // 安装会话创建成功
            Log.i("InstallProgress", "Session created: $sessionId")
        }

        override fun onProgressChanged(sessionId: Int, progress: Float) {
            // 安装进度更新
            itemList[mPosition].installed = 1
            itemList[mPosition].progress = progress * 100
            adapter.notifyItemChanged(mPosition)
            Log.d("installApk", "onProgressChanged: $mPosition ${itemList[mPosition].installed} $progress")
        }

        override fun onFinished(sessionId: Int, success: Boolean) {
            if (success) {
                // 安装成功
                itemList[mPosition].installed = 2
            } else {
                // 安装失败
                itemList[mPosition].installed = 3
            }
            adapter.notifyItemChanged(mPosition)
            Log.d("installApk", "onProgressChanged: $mPosition ${itemList[mPosition].installed} $success")
        }

        override fun onActiveChanged(sessionId: Int, active: Boolean) {
            // 安装会话激活状态发生变化
        }

        override fun onBadgingChanged(sessionId: Int) {
            // 应用图标发生变化
        }
    }



    private fun installApk(position: Int) {
        val packageInstaller = packageManager.packageInstaller
        val params =
            PackageInstaller.SessionParams(PackageInstaller.SessionParams.MODE_FULL_INSTALL)
        params.setSize(File(itemList[position].path).length())

        val sessionId = packageInstaller.createSession(params)
        val session = packageInstaller.openSession(sessionId)

        try {
            val inputStream = FileInputStream(itemList[position].path)
            val outputStream = session.openWrite("base.apk", 0, -1) // -1 表示自动计算大小
            inputStream.copyTo(outputStream)
            session.fsync(outputStream)
            inputStream.close()
            outputStream.close()

            val intent = Intent(context, PackageInstallReceiver::class.java).apply {
                action = itemList[position].pkg
                putExtra(PackageInstaller.EXTRA_PACKAGE_NAME, itemList[position].pkg)
            }

            val pendingIntent = PendingIntent.getBroadcast(
                context,
                0,
                intent,
                PendingIntent.FLAG_UPDATE_CURRENT
            )
            session.commit(pendingIntent.intentSender)
        } catch (e: IOException) {
            session.abandon()
            e.printStackTrace()
        }
    }

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2395417.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

大数据-274 Spark MLib - 基础介绍 机器学习算法 剪枝 后剪枝 ID3 C4.5 CART

点一下关注吧!!!非常感谢!!持续更新!!! 大模型篇章已经开始! 目前已经更新到了第 22 篇:大语言模型 22 - MCP 自动操作 FigmaCursor 自动设计原型 Java篇开…

力扣面试150题--二叉树的锯齿形层序遍历

Day 56 题目描述 思路 锯齿形就是一层是从左向右,一层是从右向左,那么我们可以分析样例,对于第奇数层是从左向右,第偶数层是从右向左,于是可以采取一个计数器,采取链表方式,从左向右就是正常插…

如何在 CentOS / RHEL 上修改 MySQL 默认数据目录 ?

MySQL 是一个广泛使用的开源关系数据库管理系统(RDBMS),为无数的 web 应用程序和服务提供支持。默认情况下,MySQL 将其数据存储在预定义的目录中,这可能并不总是适合您的需求。您可能希望将数据目录移动到另一个位置以获得更好的性能和安全性…

简历制作要精而不简

不得不说,不管是春招,还是秋招,我们在求职时,第一步便是制作一份简历。不得不承认,好的简历,就像一块敲门砖,能让面试官眼前一亮,让应聘成功的概率增添一分。 对于一个初次求职者来…

SPA-RL:通过Stepwise Progress Attribution训练LLM智能体

SPA-RL:通过Stepwise Progress Attribution训练LLM智能体 在大语言模型(LLM)驱动智能体发展的浪潮中,强化学习(RL)面临着延迟奖励这一关键挑战。本文提出的SPA-RL框架,通过创新的分步进度归因机…

【深度学习】9. CNN性能提升-轻量化模型专辑:SqueezeNet / MobileNet / ShuffleNet / EfficientNet

SqueezeNet / MobileNet / ShuffleNet / EfficientNet 一、背景与动机 随着深度神经网络在图像识别任务上取得巨大成功,它们的结构越来越深、参数越来越多。然而在移动端或嵌入式设备中: 存储资源有限推理计算能力弱能耗受限 因此,研究者…

Relational Algebra(数据库关系代数)

目录 What is an “Algebra” What is Relational Algebra? Core Relational Algebra Selection Projection Extended Projection Product(笛卡尔积) Theta-Join Natural Join Renaming Building Complex Expressions Sequences of Assignm…

Chorme如何对于youtube视频进行画中画背景播放?

画中画可以让你小窗播放,然后浏览器放后台还可以做点别的事情。 B站直接可以选择小窗播放,游览器最小化就可以,但是youtube的小窗播放游览器一切换就不显示了。 其实是因为youtube的小窗播放不是真的小窗播放。要想真的实现需要在youtube视…

017搜索之深度优先搜索——算法备赛

深度优先搜索 如果说广度优先搜索是逐层扩散,那深度优先搜索就是一条道走到黑。 深度优先遍历是用递归实现的,预定一条顺序规则(如上下左右顺序) ,一直往第一个方向搜索直到走到尽头或不满足要求后返回上一个叉路口按…

睿抗机器人开发者大赛CAIP-编程技能赛-历年真题 解题报告汇总 | 珂学家

前言 汇总 睿抗机器人开发者大赛CAIP-编程技能赛-历年真题 解题报告汇总 2024年 2024 睿抗机器人开发者大赛CAIP-编程技能赛-本科组 (国赛) 解题报告 2024 睿抗机器人开发者大赛CAIP-编程技能赛-本科组(省赛)解题报告 2024 睿抗机器人开发者大赛CAI…

【c++】【数据结构】AVL树

目录 AVL树的定义AVL树的部分模拟实现平衡因子的引入平衡因子的向上调整旋转算法单旋算法右单旋左单旋 双旋算法左右双旋右左双旋 AVL树的定义 AVL树本质是一种搜索二叉树,传统的二叉搜索树我们都有所了解,其在理想情况下也就是接近满二叉树时拥有极高的…

通义灵码深度实战测评:从零构建智能家居控制中枢,体验AI编程新范式

一、项目背景:零基础挑战全栈智能家居系统 目标:开发具备设备控制、环境感知、用户习惯学习的智能家居控制中枢(PythonFlaskMQTTReact) 挑战点: 需集成硬件通信(MQTT)、Web服务(Flask)、前端交互(React) 调用天气AP…

头歌之动手学人工智能-Pytorch 之优化

目录 第1关:如何使用optimizer 任务描述 编程要求 测试说明 真正的科学家应当是个幻想家;谁不是幻想家,谁就只能把自己称为实践家。 —— 巴尔扎克开始你的任务吧,祝你成功! 第2关:optim.SGD 任务描述…

基于谷歌ADK的智能客服系统简介

Google的智能体开发工具包(Agent Development Kit,简称ADK)是一个开源的、以代码为中心的Python工具包,旨在帮助开发者更轻松、更灵活地构建、评估和部署复杂的人工智能智能体(AI Agent)。ADK 是一个灵活的…

(一)视觉——工业相机(以海康威视为例)

一、工业相机介绍 工业相机是机器视觉系统中的一个关键组件,其最本质的功能就是将光信号转变成有序的电信号。选择合适的相机也是机器视觉系统设计中的重要环节,相机的选择不仅直接决定所采集到的图像分辨率、图像质量等,同时也与整个系统的运…

DAY 36 超大力王爱学Python

仔细回顾一下神经网络到目前的内容,没跟上进度的同学补一下进度。 作业:对之前的信贷项目,利用神经网络训练下,尝试用到目前的知识点让代码更加规范和美观。探索性作业(随意完成):尝试进入nn.Mo…

SRD-12VDC-SL-C 继电器‌接线图解

这个继电器可以使用12伏的直流电源控制250伏和125伏的交流电,也可以控制30伏和28伏的直流电,电流都为10安。 此继电器有5个引脚,各个的作用如下: 引脚4和引脚5为触点, 引脚1和引脚3为线圈引脚,接12伏的直…

基于开源链动2+1模式AI智能名片S2B2C商城小程序的企业组织生态化重构研究

摘要:本文以互联网时代企业组织结构变革为背景,探讨开源链动21模式AI智能名片S2B2C商城小程序在推动企业从封闭式向开放式生态转型中的核心作用。通过分析传统企业资源获取模式与网络化组织生态的差异,结合开源链动21模式的裂变机制、AI智能名…

2,QT-Creator工具创建新项目教程

目录 1,创建一个新项目 demo_01.pro(项目配置文件) 类似 CMakeList.txt widget.h(头文件)​ main.cpp(程序入口)​ widget.cpp(源文件)​ widget.ui(界面设计文件)​ 1,创建一个新项目 依次选择: 设置路径: 选择编译器: 如果选择CMake, 就会生成cmakel…

《深入解析SPI协议及其FPGA高效实现》-- 第一篇:SPI协议基础与工作机制

第一篇:SPI协议基础与工作机制 1. 串行外设接口导论 1.1 SPI的核心定位 协议本质 : 全双工同步串行协议(对比UART异步、IC半双工)核心优势 : 无寻址开销(通过片选直连)时钟速率可达100MHz&…