当前位置: 首页 > article >正文

Android 12+ 上 NetworkStatsManager 统计应用流量,为什么你的 queryDetailsForUid 总返回0?

Android 12 流量统计实战破解 NetworkStatsManager.queryDetailsForUid 返回0的迷局在开发流量监控类应用时许多开发者都会遇到一个令人抓狂的问题明明按照官方文档调用了queryDetailsForUid方法却总是得到0值返回。这就像在黑暗房间里寻找开关——你知道它应该在那里但就是摸不到。本文将带你深入Android流量统计机制的核心揭示那些官方文档没有明确说明的潜规则。1. 理解NetworkStatsManager的两种查询模式Android的流量统计API看似简单实则暗藏玄机。关键在于理解系统如何处理和存储网络使用数据。1.1 历史查询 vs 摘要查询NetworkStatsManager提供了两种根本不同的数据查询方式历史查询(History queries)包括queryDetails和queryDetailsForUid等方法特点数据按小时粒度存储只统计已完成的计费周期返回原始数据记录摘要查询(Summary queries)包括querySummary和querySummaryForDevice等方法特点实时聚合数据包含当前未完成周期的统计返回汇总后的结果// 历史查询示例 - 可能返回0 NetworkStats stats manager.queryDetailsForUid( ConnectivityManager.TYPE_MOBILE, null, startTime, endTime, targetUid ); // 摘要查询示例 - 通常能获取实时数据 NetworkStats stats manager.querySummary( ConnectivityManager.TYPE_MOBILE, null, startTime, endTime );1.2 系统数据收集机制Android不会实时记录每一条网络请求而是采用批处理方式系统定期(约每小时)将网络使用数据写入持久化存储只有完成的数据周期才会被历史查询方法获取当前周期的数据只能通过摘要查询获取提示这就是为什么你刚刷完抖音立即查询却得到0 - 数据还未被系统持久化2. queryDetailsForUid返回0的六大原因及解决方案经过对数十个真实案例的分析我们总结出以下常见陷阱2.1 时间窗口设置不当问题现象查询当天数据返回0但查询昨天数据正常根本原因历史查询只统计完整的小时段解决方案// 错误示范 - 查询当天未完成时段 long start getStartOfToday(); // 今天00:00 long end System.currentTimeMillis(); // 现在 // 正确做法1 - 查询已完成的昨天 long start getStartOfYesterday(); long end getStartOfToday(); // 正确做法2 - 扩大时间范围到未来 long end getStartOfTomorrow(); // 包含今天未完成时段2.2 未正确处理subscriberId参数问题现象WiFi数据正常但移动数据总是0关键发现Android 10版本中传空字符串()查询失败传null查询成功// 危险代码 String subscriberId ; // 将导致移动数据查询失败 // 推荐写法 String subscriberId Build.VERSION.SDK_INT Build.VERSION_CODES.Q ? null : getSubscriberId();2.3 权限配置不全即使声明了权限仍可能因权限不足返回0!-- 必须声明的权限 -- uses-permission android:nameandroid.permission.READ_PHONE_STATE/ uses-permission android:nameandroid.permission.PACKAGE_USAGE_STATS tools:ignoreProtectedPermissions/额外要求PACKAGE_USAGE_STATS需要用户手动授权在Android 11上需要添加声明queries intent action android:nameandroid.intent.action.MAIN / /intent /queries2.4 UID获取方式错误常见错误获取UID的方式错误方法问题描述正确替代PackageManager.getApplicationInfo().uid可能返回错误UIDBinder.getCallingUid()硬编码测试UID不同设备UID不同动态查询// 可靠获取当前应用UID的方法 int uid Process.myUid(); // 获取其他应用UID的正确方式 ApplicationInfo ai pm.getApplicationInfo(packageName, 0); int targetUid ai.uid;2.5 网络类型选择错误典型错误只查询TYPE_MOBILE忽略WiFi使用过时的网络类型常量现代Android设备应检查的网络类型// 基本网络类型 ConnectivityManager.TYPE_WIFI ConnectivityManager.TYPE_MOBILE // 可能需要考虑的类型 ConnectivityManager.TYPE_ETHERNET ConnectivityManager.TYPE_VPN2.6 未处理多用户情况在支持多用户的设备上每个用户有独立的UID空间默认只查询当前用户数据需要特殊权限查询其他用户数据解决方案if (Build.VERSION.SDK_INT Build.VERSION_CODES.P) { // 使用createForAllProfiles参数 NetworkStatsManager manager context.getSystemService( NetworkStatsManager.class ); }3. 实战构建健壮的流量监控工具类基于以上经验我们设计一个可靠的流量统计工具3.1 核心实现代码public class NetworkStatsHelper { private static final String TAG NetworkStatsHelper; /** * 获取应用网络使用数据兼容历史查询和摘要查询 */ public static NetworkUsage getAppUsage(Context context, int uid, long startTime, long endTime) { NetworkStatsManager manager (NetworkStatsManager) context.getSystemService(Context.NETWORK_STATS_SERVICE); NetworkUsage usage new NetworkUsage(); // 先尝试摘要查询获取实时数据 try { addSummaryUsage(manager, ConnectivityManager.TYPE_WIFI, null, startTime, endTime, uid, usage); addSummaryUsage(manager, ConnectivityManager.TYPE_MOBILE, null, startTime, endTime, uid, usage); } catch (Exception e) { Log.w(TAG, Summary query failed, fallback to details, e); } // 如果数据为0尝试历史查询 if (usage.totalBytes() 0) { try { addDetailsUsage(manager, ConnectivityManager.TYPE_WIFI, null, startTime, endTime, uid, usage); addDetailsUsage(manager, ConnectivityManager.TYPE_MOBILE, null, startTime, endTime, uid, usage); } catch (Exception e) { Log.e(TAG, Details query also failed, e); } } return usage; } private static void addSummaryUsage(NetworkStatsManager manager, int networkType, String subscriberId, long start, long end, int uid, NetworkUsage out) { NetworkStats stats manager.querySummary( networkType, subscriberId, start, end ); try { NetworkStats.Bucket bucket new NetworkStats.Bucket(); while (stats.hasNextBucket()) { stats.getNextBucket(bucket); if (bucket.getUid() uid) { out.rxBytes bucket.getRxBytes(); out.txBytes bucket.getTxBytes(); } } } finally { stats.close(); } } // 类似实现addDetailsUsage... }3.2 使用示例// 获取抖音的流量使用过去24小时 String packageName com.ss.android.ugc.aweme; // 抖音包名 ApplicationInfo ai pm.getApplicationInfo(packageName, 0); int uid ai.uid; long end System.currentTimeMillis(); long start end - 24 * 60 * 60 * 1000; // 24小时前 NetworkUsage usage NetworkStatsHelper.getAppUsage( context, uid, start, end ); Log.d(Traffic, String.format(抖音使用: 上传%.2fMB, 下载%.2fMB, usage.txBytes / (1024f * 1024f), usage.rxBytes / (1024f * 1024f)));3.3 性能优化建议缓存查询结果历史数据不会频繁变化可适当缓存批量查询避免频繁单独查询每个应用后台服务定期更新数据而非实时查询// 批量查询优化示例 MapInteger, NetworkUsage batchQuery(ListInteger uids, long start, long end) { NetworkStats stats manager.querySummary( ConnectivityManager.TYPE_MOBILE | ConnectivityManager.TYPE_WIFI, null, start, end ); MapInteger, NetworkUsage result new HashMap(); NetworkStats.Bucket bucket new NetworkStats.Bucket(); while (stats.hasNextBucket()) { stats.getNextBucket(bucket); int uid bucket.getUid(); if (uids.contains(uid)) { NetworkUsage usage result.getOrDefault(uid, new NetworkUsage()); usage.rxBytes bucket.getRxBytes(); usage.txBytes bucket.getTxBytes(); result.put(uid, usage); } } stats.close(); return result; }4. 高级技巧与疑难解答4.1 处理Android 12的新限制Android 12引入了更严格的限制后台限制应用在后台时查询可能失败缓存失效系统可能清除旧统计数据权限变更需要额外声明权限适配方案// 检查后台限制 if (Build.VERSION.SDK_INT Build.VERSION_CODES.S) { ActivityManager am (ActivityManager) context.getSystemService( Context.ACTIVITY_SERVICE); if (!am.isBackgroundRestricted()) { // 执行查询 } }4.2 调试技巧当查询仍然返回0时检查系统日志adb logcat | grep -E NetworkStats|netd验证权限状态// 检查PACKAGE_USAGE_STATS权限 AppOpsManager appOps (AppOpsManager) context.getSystemService(Context.APP_OPS_SERVICE); int mode appOps.checkOpNoThrow( AppOpsManager.OPSTR_GET_USAGE_STATS, Process.myUid(), context.getPackageName() ); boolean granted mode AppOpsManager.MODE_ALLOWED;交叉验证// 使用TrafficStats作为参考 long mobileRx TrafficStats.getMobileRxBytes(); long mobileTx TrafficStats.getMobileTxBytes();4.3 厂商定制系统适配不同厂商可能有特殊处理厂商已知问题解决方案小米可能限制后台查询加入电池优化白名单华为需要额外权限申请华为移动服务权限OPPO自动清理统计数据缩短查询间隔// 通用厂商适配检查 if (isXiaomiDevice()) { if (!isIgnoringBatteryOptimizations(context)) { Intent intent new Intent( Settings.ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS); intent.setData(Uri.parse(package: context.getPackageName())); context.startActivity(intent); } }在华为设备上开发时第一次遇到查询返回0的情况我花了整整两天时间才发现需要在华为应用市场额外声明权限。这个教训让我明白在Android生态中有时候官方文档只是故事的开始。

相关文章:

Android 12+ 上 NetworkStatsManager 统计应用流量,为什么你的 queryDetailsForUid 总返回0?

Android 12 流量统计实战:破解 NetworkStatsManager.queryDetailsForUid 返回0的迷局 在开发流量监控类应用时,许多开发者都会遇到一个令人抓狂的问题:明明按照官方文档调用了 queryDetailsForUid 方法,却总是得到0值返回。这就像…...

ST7789V SPI 4线接口LCD屏驱动实战:从硬件连接到完整初始化代码

ST7789V SPI 4线接口LCD屏驱动实战:从硬件连接到完整初始化代码 在嵌入式开发中,LCD显示屏作为人机交互的重要组件,其驱动实现一直是开发者关注的焦点。ST7789V作为一款广泛应用于中小尺寸LCD屏的驱动IC,以其出色的色彩表现和灵活…...

MQTTX+Qt联合调试指南:手把手搭建物联网通信测试环境

MQTTXQt联合调试指南:手把手搭建物联网通信测试环境 在物联网开发中,MQTT协议因其轻量级和高效性成为设备通信的首选方案。而Qt框架的跨平台特性与MQTTX工具的直观可视化界面,为开发者提供了从原型验证到产品落地的完整工具链。本文将带您从零…...

计算机网络的计算模式

计算模式指的是网络中计算任务(数据处理、存储、运算等)在客户端和服务器之间如何分配与协作。随着技术发展,主要经历了以下几种模式的演变。一、计算模式的主要类型模式核心特点处理位置典型代表集中式计算模式所有计算在主机完成&#xff0…...

Qt文件操作避坑指南:QFile与QTextStream/QDataStream的最佳搭配方案

Qt文件操作避坑指南:QFile与QTextStream/QDataStream的最佳搭配方案 在Qt开发中,文件操作是每个开发者都会遇到的基础需求。无论是配置文件读写、数据持久化还是日志记录,都离不开对文件系统的操作。Qt提供了QFile、QTextStream和QDataStream…...

ESP32 OTA升级实战:从官方native_ota_example到自定义固件服务器的完整配置指南

ESP32 OTA升级实战:从官方示例到生产级部署的进阶指南 当你的ESP32设备部署在远程现场,每次更新固件都要派人去现场烧录?这种低效方式早已过时。OTA(Over-The-Air)技术让设备像智能手机一样远程更新,而ESP3…...

CVAT在Ubuntu 20.04上的完整安装指南:从Docker配置到多人协作避坑

CVAT在Ubuntu 20.04上的完整安装指南:从Docker配置到多人协作避坑 在计算机视觉项目中,高质量的数据标注是模型成功的关键。CVAT(Computer Vision Annotation Tool)作为英特尔开源的图像标注工具,凭借其丰富的标注功能…...

TwinCAT3 ADS路由死活加不上?别慌,这份保姆级排查清单帮你搞定(附Win7/CE系统差异)

TwinCAT3 ADS路由添加失败全场景排查指南:从原理到实战 想象一下这样的场景:凌晨两点的生产线突然停机,你顶着黑眼圈站在控制柜前,TwinCAT3的ADS路由死活加不上——这种时候需要的不是教科书式的理论,而是能快速定位问…...

【AGI时代招聘生存指南】:错过2026奇点大会这4个信号,你的技术团队将在6个月内掉队2个代际

第一章:2026奇点智能技术大会:AGI与人才招聘 2026奇点智能技术大会(https://ml-summit.org) AGI招聘范式的结构性转变 传统技术岗位JD正被AGI原生能力模型重构。企业不再仅评估编程语言熟练度,而是聚焦于候选人在多模态推理、自主目标分解、…...

别再只用get()了!Java Stream中filter+findAny的3种安全写法与避坑指南

别再只用get()了!Java Stream中filterfindAny的3种安全写法与避坑指南 在日常Java开发中,我们经常需要从集合中查找满足特定条件的元素。Stream API的filter和findAny组合看似简单,但直接使用get()方法却隐藏着不小的风险。本文将带你深入理解…...

Windows 11 先装,Arch Linux 后装:UEFI 双系统启动菜单避坑全记录

Windows 11 与 Arch Linux 双系统 UEFI 引导完全避坑指南 每次看到论坛里有人抱怨"装完双系统找不到启动菜单",我就想起自己第一次尝试时的狼狈经历。那天深夜,我对着黑屏反复重启了十七次,最终在凌晨三点意识到问题出在一个看似微…...

diff-pdf终极指南:3分钟学会PDF视觉差异比对,让文档修改无所遁形

diff-pdf终极指南:3分钟学会PDF视觉差异比对,让文档修改无所遁形 【免费下载链接】diff-pdf A simple tool for visually comparing two PDF files 项目地址: https://gitcode.com/gh_mirrors/di/diff-pdf 你是否曾花费数小时对比两个版本的PDF文…...

AzurLaneAutoScript技术架构深度解析:构建碧蓝航线7x24小时智能自动化系统

AzurLaneAutoScript技术架构深度解析:构建碧蓝航线7x24小时智能自动化系统 【免费下载链接】AzurLaneAutoScript Azur Lane bot (CN/EN/JP/TW) 碧蓝航线脚本 | 无缝委托科研,全自动大世界 项目地址: https://gitcode.com/gh_mirrors/az/AzurLaneAutoSc…...

AI教材写作大揭秘:实用工具推荐,助力低查重教材快速编写!

传统资料整合困境与AI写教材的优势 编写教材离不开丰富的资料支持,但传统的资料整合方式已经难以满足我们日益增长的需求。过去,想要从课程标准、学术文献、教学案例中提炼出有价值的信息,得在知网、教研平台等各个渠道间费时费力&#xff0…...

终极指南:如何快速掌握Unity游戏逆向工程利器Il2CppDumper

终极指南:如何快速掌握Unity游戏逆向工程利器Il2CppDumper 【免费下载链接】Il2CppDumper Unity il2cpp reverse engineer 项目地址: https://gitcode.com/gh_mirrors/il/Il2CppDumper 想要深入了解Unity游戏内部机制吗?Il2CppDumper 是当前最强大…...

2025届学术党必备的降AI率工具推荐

Ai论文网站排名(开题报告、文献综述、降aigc率、降重综合对比) TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 为了降低文本的 AIGC 率,得从语言自然度与结构差异性这两个关键要点着手。就语言…...

3分钟掌握Windows三指拖拽:让触控板操作效率翻倍

3分钟掌握Windows三指拖拽:让触控板操作效率翻倍 【免费下载链接】ThreeFingersDragOnWindows Enables macOS-style three-finger dragging functionality on Windows Precision touchpads. 项目地址: https://gitcode.com/gh_mirrors/th/ThreeFingersDragOnWindo…...

数据提取革命:如何用WebPlotDigitizer从图表中解放数值宝藏

数据提取革命:如何用WebPlotDigitizer从图表中解放数值宝藏 【免费下载链接】WebPlotDigitizer Computer vision assisted tool to extract numerical data from plot images. 项目地址: https://gitcode.com/gh_mirrors/we/WebPlotDigitizer 你是否曾面对学…...

5分钟掌握Python剪映API:让视频剪辑效率提升10倍的终极指南

5分钟掌握Python剪映API:让视频剪辑效率提升10倍的终极指南 【免费下载链接】JianYingApi Third Party JianYing Api. 第三方剪映Api 项目地址: https://gitcode.com/gh_mirrors/ji/JianYingApi 你是否厌倦了每天重复的视频剪辑工作?添加水印、调…...

混音教学第五课|从零认识 RVC:软件启动全流程真机实操(GTX1050Ti 专属)

作者:龙沅可 各位音乐编程圈的兄弟,我是深耕实战 3 年的地下程序员胡桃。前面我们走完了人声分离、软件模型全套准备、Anaconda 环境兜底、VOCALOID&RVC 选择杂谈、官方作品技术复盘 个人修复版全流程,本期终于回归主线实操,…...

Windows 11系统清理优化终极指南:使用Win11Debloat提升50%性能

Windows 11系统清理优化终极指南:使用Win11Debloat提升50%性能 【免费下载链接】Win11Debloat A simple, lightweight PowerShell script that allows you to remove pre-installed apps, disable telemetry, as well as perform various other changes to declutte…...

WebLaTeX:在线LaTeX编辑新体验,告别繁琐配置的写作利器

WebLaTeX:在线LaTeX编辑新体验,告别繁琐配置的写作利器 【免费下载链接】WebLaTex A complete alternative for Overleaf with VSCode Web Git Integration Copilot Grammar & Spell Checker Live Collaboration Support. Based on GitHub Code…...

Godot-MCP:重构游戏开发效率的AI协作框架解决方案

Godot-MCP:重构游戏开发效率的AI协作框架解决方案 【免费下载链接】Godot-MCP An MCP for Godot that lets you create and edit games in the Godot game engine with tools like Claude 项目地址: https://gitcode.com/gh_mirrors/god/Godot-MCP 传统游戏开…...

Vue v-on 在 React 中 VuReact 会如何实现?

VuReact 是一个能将 Vue 3 代码编译为标准、可维护 React 代码的工具。今天就带大家直击核心:Vue 中常见的 v-on/ 指令经过 VuReact 编译后会变成什么样的 React 代码? 前置约定 为避免示例代码冗余导致理解偏差,先明确两个小约定&#xff…...

Vue v-bind 转 React:VuReact 怎么处理?

VuReact 是一个能将 Vue 3 代码编译为标准、可维护 React 代码的工具。今天就带大家直击核心:Vue 中常见的 v-bind/: 指令经过 VuReact 编译后会变成什么样的 React 代码? 前置约定 为避免示例代码冗余导致理解偏差,先明确两个小约定&#…...

IDEA2024实战:两种主流方式搭建Maven Web项目(附避坑指南)

1. 两种主流方式搭建Maven Web项目概述 在IDEA2024中创建Maven Web项目,主要有两种主流方式:使用Archetype骨架和手动配置Web模块。这两种方式各有优缺点,适用于不同的开发场景。作为一个长期使用IDEA进行Java Web开发的程序员,我…...

Java的java.util.HexFormat双向支持

Java 16引入的java.util.HexFormat类为开发者提供了高效的十六进制与二进制数据双向转换能力,填补了Java标准库在十六进制处理领域的空白。这个工具类不仅支持基础格式转换,还能处理字节数组、字符序列等复杂场景,其线程安全特性更使其成为网…...

技术工厂中的对象生产与配置管理

技术工厂中的对象生产与配置管理 在数字化与智能化快速发展的今天,技术工厂已成为现代工业的核心。对象生产与配置管理作为技术工厂的关键环节,直接影响生产效率、资源利用率和产品质量。通过智能化的对象生产与动态化的配置管理,企业能够实…...

Rust 枚举与模式匹配的高级用法

Rust 枚举与模式匹配的高级用法 Rust 的枚举(Enum)和模式匹配(Pattern Matching)是其强大类型系统的核心特性之一,不仅能够简洁地表达复杂的数据结构,还能通过模式匹配实现精准的逻辑控制。对于已经掌握基…...

技术分享的有效组织与演讲技巧提升方法

技术分享的有效组织与演讲技巧提升方法 在技术领域,分享知识与经验是推动团队成长的重要方式。如何将复杂的技术内容清晰传达,并吸引听众的注意力,是许多技术从业者面临的挑战。本文将探讨技术分享的有效组织方法,并分享提升演讲…...