[授权发表]Android加载不同DPI资源与内存消耗之间的关系

2023-05-16

by Will of TinyLab.org
2015/04/21

最初发表:泰晓科技 – 聚焦嵌入式 Linux,追本溯源,见微知著!
原文链接:Android加载不同DPI资源与内存消耗之间的关系
评论说明:为更好地聚合大家的讨论,请到上面原文的评论区回复。


Android DPI 分级标准简介

Android 设备在物理尺寸和屏幕密度上都有很大的不同,为了简化多设备的设计方案,就是设定一套分级标准。屏幕密度上的分级标准就是:LDPI、MDPI、HDPI、XHDPI,也就是各种大小的 DPI(Dots per inch)。DPI 就是屏幕像素密度的衡量标准

不同设备共享同一套 DPI 资源有哪些问题?

现在进入正题。

Q:不少公司出于简化设计和研发的目的,往往在方案中只使用一套 DPI 资源,这样做可不可行呢?

A:Android 有一套加载资源的规则,如果对应的 DPI 文件夹不存在要用的资源就会按照规则去找其它的DPI 文件夹,如果最终能找到就可以使用。所以上述方案是“可行的”-- 可以正常运行不报错的。那可行性的另一个方面就是对性能有没有影响。上述问题就变为下面问题:

Q:同一套 DPI 资源在不同手机上使用时内存消耗有什么不同? 或 App 中加载不同 DPI 文件夹中的资源内存消耗有什么不同?

问题:DPI 越小的文件夹内存消耗越大?

下面以 png 图片的加载为例。

demo pic

原始图片(attribute—width:960,height:540,bit depth:32,size:217082bytes)。

做简单的 demo app,即在 activity 中只加载这一个图片。

放在 hdpi 文件夹中,dumpsys meminfo 后发现 Heap Alloc 为 5420,远远大于 size,所以先肯定的是内存消耗与图片文件大小无关。

再放到不同的 DPI 文件夹中发现:越是 DPI 小的文件夹内存消耗越大!

分析:加载低 DPI 资源会额外拉伸放大图片

由于 Heap Alloc 只能看到堆的分配总体大小,不能看到上述发现有什么“规律”,所以接着使用 MAT 分析。

在 hdpi 中抓取 hprof 文件,用 MAT 打开:

mat

见图中的 byte 数组,大小为 2073600,这个大小就是加载的那张 png 图片占用的内存大小。

分别分析图片资源放在 mdpi、ldpi 和 xhdpi 时的 hprof 文件,byte 数组大小分别为:4665600、8294400、1166400。不同 DPI 文件夹与图片占用的内存大小关系如下:

DPIsldpimdpihdpixhdpi
Byte[] size8294400466560020736001166400
Ratio8^26^24^23^2

开始就说到 Android 的屏幕密度的分级标准是 LDPI、MDPI、HDPI、XHDPI 这些各种大小的 DPI。也就是 LDPI 的设备默认使用的是 ldpi 文件夹下的资源。根据 DPI 值的大小再整理一下,屏幕像素密度的值对应使用的 DPI 文件夹关系如下:

DPIsldpimdpihdpixhdpi
Density120160240320
Ratio3468

根据上面两个表格的 Ratio 值,可以发现内存占用和 DPI 资源是有一定规律的。其实我们知道 png 加载内存的消耗与文件大小无关,而是与 png 图片的长宽和位深有关,也就是:


Memory Consumption Size(UOM:byte) = Width * Height * (Bit depth / 8)
  

上面公式是不能完全在 Android 中使用的,根据上述找到的规律,Android 中 png 图片内存消耗公式可以概括为:


Memory Consumption Size(UOM:byte) = ScaledWidth * ScaledHeight * (Bit depth / 8)
ScaledWidth = Width * factor
ScaledHeight = Height * factor
factor = DENSITY_DEVICE / ResourceDensity  // DENSITY_DEVICE 是设备的 DPI 大小, ResourceDensity 是设备加载的 DPI 文件夹对应的 DPI 大小
  

所以:


Memory Consumption Size = Width * Height * (DENSITY_DEVICE / ResourceDensity)^2 * (Bit depth / 8)
  

上述 hdpi 中的 2073600 可以由此计算得出:


960 * 540 * (240 / 240)^2 * (32 / 8) = 2073600
  

在 BitmapFactory.cpp 的 doDecode() 中 添加 log ,可验证上述公式(资源在 xdpi 中,sx、sy 就是上述公式中的 factor):


01-15 21:00:49.479  3079  3079 D BitmapFactory: doDecode----sx:0.750000 ,sy:0.750000 ,scaledWidth:405 ,scaledHeight:720 ,decodingBitmap.width:540 ,decodingBitmap.heigth:960
  

Android 加载资源默认选用和设备 DPI 匹配的资源,如果没有就去到其它 DPI 文件夹中寻找资源。找到后它会认为使用了不同 DPI 的资源,为了保持与设备 DPI 一致,就会对资源做拉伸或缩放处理再加载。下面是上述 png 图片分别放在 mdpi 和 xxhdpi 文件夹下的截图:

mdpixxhdpi
mdpixxhdpi

很明显就可以看到在 xxhdpi 下时的截图模糊了不少。使用的测试手机是 hdpi 的,但是默认 hdpi 找不到图片资源,它就会按照一定规则找到我放在 xxhdpi 中的资源。手机认为从 xxhdpi 获取的资源比手机的 dpi 要高,它就会按照表格中的比例把资源缩小,也就是加载到内存中的图片资源已经是原来大小的 1/2,占用的内存当然会缩小不做缩放操作图片的 1/4。但是坏处也是显而易见的,显示到手机的图片资源清晰度下降,模糊了很多。

相反的,hdpi 的手机加载低 dpi 资源,例如 ldpi,加载到内存前会先按比例拉伸。拉伸后再显示到手机中清晰度是没有问题,但是内存占用确增大为原来的 4 倍!还是要注意到这一点,如果图片资源在 app 中放错 dpi 文件夹,使用体验会大打折扣,或者尽量使用 9patch 图片。

小结:建议根据设备配置 DPI 资源

现在就可以回答提出的问题了:

Q:同一套 DPI 资源在不同手机上使用时内存消耗有什么不同? 或 App 中加载不同 DPI 文件夹中的资源内存消耗有什么不同?

A:不要使用一套资源适用于各种不同 DPI 的设备,这样图片的清晰度和内存消耗都会有问题。这就是为什么 Android 要求对不同 DPI 文件做不同的资源,并且不同 DPI 资源的长宽比要与 DPI Ratio 相对应。

PS:此文结论一句话就能给说清楚,但推导的过程更重要。爱因斯坦说过:Imagination is more important than knowledge.

参考资料

参考的 Android 源码:


/frameworks/base/core/java/android/util/DisplayMetrics.java 
/frameworks/base/graphics/java/android/graphics/Bitmap.java
/frameworks/base/graphics/java/android/graphics/drawable/BitmapDrawable.java
/frameworks/base/core/jni/android/graphics/BitmapFactory.cpp
  
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

[授权发表]Android加载不同DPI资源与内存消耗之间的关系 的相关文章

  • 飞行机器人专栏(十一)-- 空中机器人综合健康管理系统

    目录 一 综合健康管理系统架构 1 系统架构 xff1a 2 故障诊断算法 xff1a 3 预测维护策略 xff1a 4 安全与隐私 xff1a 5 用户友好性 xff1a 6 模块化与可扩展性 xff1a 7 与其他系统集成 xff1a
  • 飞行机器人专栏(十二)-- 提高机器人系统可靠性的关键要素与实践

    本文将介绍如何在机器人系统的开发过程中融入关键要素 xff0c 从而提高系统的可靠性 我们将从需求分析 设计阶段 开发与调试 验证与优化 迭代与升级等方面进行详细讨论 xff0c 并提供示例代码以帮助您更好地理解相关概念 目录 一 需求分析
  • GPT 学术优化版使用指南 -- GPT Academic

    nbsp 目录 1 项目介绍 1 1 简介 nbsp 1 2 功能说明 2 环境配置 2 1 本地安装
  • CvBridge在ROS图像和OpenCV图像之间进行转化

    目录 1 概念 2 将ROS图像转化为OpenCV图像 3 将OpenCV图像转化为ROS图像消息 4 例子 1 概念 ROS的 sensor msgs Image消息格式本身就是为了传递图像 xff0c 但是因为因为不能直接传递给Open
  • VINS_Fusion回环检测学习笔记

    回环检测的任务主要是为了检测机器人是否到达之前相同的位置 xff0c 并且消累计的误差 VINS Fusion的回环检测和VINS Mono基本相似 只是loop fusion中读取的很多参数是直接设定好的 回环部分我们先从pose gra
  • STM32——ST-Link v2调试下载器的连接方法【详解】

    硬件电路 在网上购买了STM32F103C8T6的最小系统开发板 xff0c 由于手头上没有官方的STM32调试器 xff0c 所以也同时购买了一个便宜的st link v2下载调试器 这种下载器和STM32的最小系统板之间完成调试通信需要
  • Linux/Centos Makefile 使用总结

    1 Makefile 简介 Makefile 是和 make 命令一起配合使用的 很多大型项目的编译都是通过 Makefile 来组织的 如果没有 Makefile 那很多项目中各种库和代码之间的依赖关系不知会多复杂 Makefile的组织
  • 深入理解SQLite3之sqlite3_exec及回调函数

    sqlite3的C C 43 43 接口API主要有3个重要函数 xff0c 分别为 1 sqlite3 open const char filename sqlite3 ppDb 2 int sqlite3 exec sqlite3 An
  • 5G:三大场景--- eMBB、URLLC、mMTC

    背景 xff1a 很多人认为 5G 确实是未来的发展方向 xff0c 但具体到哪些落地 xff0c 又说不清楚 xff0c 甚至于认为 5G 只比 4G 多了一个G 而已 xff0c 但笔者认为 xff1a 5G 在移动通信领域绝对是革命性
  • Linux:lspci命令介绍

    lspci xff0c 是一个用来显示系统中所有PCI总线设备或连接到该总线上的所有设备的工具 pci是一种总线 xff0c 而通过pci总线连接的设备就是pci设备了 如今 xff0c 我们常用的设备很多都是采用pci总线了 xff0c
  • iperf详细使用方法

    Iperf 是一个网络性能测试工具 Iperf可以测试TCP和UDP带宽质量 Iperf可以测量最大TCP带宽 xff0c 具有多种参数和UDP特性 Iperf可以报告带宽 xff0c 延迟抖动和数据包丢失 Iperf使用方法与参数说明 参
  • IPSec浅见

    1 IPSEC协议簇安全框架 a IPSec简介 IPSec xff08 Internet Protocol Security xff09 xff1a 是一组基于网络层的 xff0c 应用密码学的安全通信协议族 IPSec不是具体指哪个协议
  • IPsec:strongswan与vpp实现ipsec

    1 strongswan 43 vpp简介 strongswan与vpp如何结合 本次实验使用的是VPP 20 01 版本 43 strongswan 5 9 6版本 目前strongSwan 43 vpp的方案主要是使用strongswa
  • Linux:启动sshd服务的时候提示错误Unsupported option UsePAM

    问题 分析 默认的configure 没有启用 with pam选项 xff0c 如果在sshd config配置文件里加入UsePAM no 就会导致上面的错误提示 UsePAM与ssh密码认证相关 xff0c 但公司服务器禁止通过密码认
  • Ubuntu: ssh升级后服务不稳定不断重启,查看sshd服务状态为activating(start)的解决办法

    现象 xff1a Ubuntu20 04 升级ssh7 4到8 1版本后 xff0c ssh连接不稳定 xff0c 时断时续 xff0c systemctl status sshd查看服务状态为activating start xff0c
  • Linux:grep命令检索文件内容详解

    前言 Linux系统中搜索 查找文件中的内容 xff0c 一般最常用的是grep命令 xff0c 另外还有egrep命令 xff0c 同时vi命令也支持文件内容检索 下面来一起看看Linux利用grep命令检索文件内容的详细介绍 方法如下
  • stm32零基础应该怎么入门?

    单片机 xff08 microcontrollers xff09 是一种集成电路芯片 xff0c 是采用超大规模集成电路技术把具有数据处理能力的中央处理器CPU 多种I O口和中断系统 定时器 计数器等功能集成到一块硅片上构成的一个小而完善
  • Linux:CPU频率调节模式以及降频方法简介

    概述 cpufreq的核心功能 xff0c 是通过调整CPU的电压和频率 xff0c 来兼顾系统的性能和功耗 在不需要高性能时 xff0c 降低电压和频率 xff0c 以降低功耗 xff1b 在需要高性能时 xff0c 提高电压和频率 xf
  • Linux:rsyslog 日志丢失 messages lost due to rate-limiting

    系统日志显示 cat var log messages Apr 7 16 20 01 ngnodeb rsyslogd imjournal 154664 messages lost due to rate limiting 解决方法 修改配
  • Linux:shell 中的单行注释和多行注释

    关于 shell 中的单行注释和多行注释 单行注释 众所周知 xff0c 使用 比如想要注释 echo 34 Hello World 34 root 64 test vim test sh echo 34 Hello World 34 多行

随机推荐

  • Shell三剑客之sed:修改 xml

    修改前 vim config xml lt config input type verify 61 34 bool 34 name 61 34 flow bypass class 34 visible 61 34 true 34 gt fa
  • STM32串口通信

    STM32串口通信 一 基于寄存器与基于固件库编写的差异二 stm32串口通信实战1 烧录方式2 代码及效果图 三 C语言程序里全局变量 局部变量 堆 栈等概念四 stm32的堆 栈 全局变量的分配地址 一 基于寄存器与基于固件库编写的差异
  • keil下的FreeRtos多任务程序

    keil下的Freertos多任务程序 1 手动移植FreeRtos xff08 以STM32F103为例 xff09 2 直接使用野火的模板 1 手动移植FreeRtos xff08 以STM32F103为例 xff09 用该链接下载Fr
  • 随笔小记(二十七)

    神经网络中Epoch Iteration Batchsize相关理解和说明 batchsize xff1a 中文翻译为批大小 xff08 批尺寸 xff09 简单点说 xff0c 批量大小将决定我们一次训练的样本数目 batch size将
  • 手把手教物体检测——EfficientDet

    目录 摘要 训练数据 1 下载Pytoch版的EfficientDet 2 制作数据集 3 下载EfficientNets预训练模型 4 安装模型需要的包 5 放置数据集 6 修改train py中的参数 测试 注意 摘要 谷歌大脑团队 Q
  • 简化的围棋棋子规则(C++实现)

    题目 xff1a 输入棋盘 xff1a 1 1 2 3 2 3 3 3 2 3 3 3 2 2 2 3 3 3 1 2 2 2 3 3 2 1 1 2 3 1 其中1代表空 xff0c 2代表白子 xff0c 3代表黑子 xff09 输出
  • MATLAB中将图像转换为二值图像im2bw

    在MATLAB中将图像转换为二值图像 xff0c 主要运用im2bw函数 xff0c 涉及到一个灰度门槛的数值 对于灰度图像 bw 61 im2bw I level level空着的话 xff0c 默认是0 5 level一般使用grayt
  • Ubuntu 20.04 Gazebo安装 及模型库下载

    安装参考自官方教程noetic版本 xff0c 为了安装模型库 xff0c 就一起编辑了 1 设置你的电脑来接收软件 sudo sh c 39 echo 34 deb http packages osrfoundation org gaze
  • git pull强制覆盖本地修改

    有时本地代码做了修改 xff0c 但又想放弃这部分修改 xff0c 重新在新代码基础上进行开发 xff0c 这时可用如下方法覆盖先前修改 xff0c 并拉取远程仓更新本地代码 方法一 xff1a git fetch git reset ha
  • gazebo中视觉仿真怎么使用自定义贴图的问题

    gazebo中提供了很少的贴图 xff0c 场景只是用这几张贴图 xff0c 视觉SLAM仿真很容易在不该闭环的时候闭环 xff0c 导致根本没法用 那么我们怎么添加自己的贴图呢 xff1f 首先gazebo建模 使用默认贴图 xff0c
  • 传统定位方法简介--------里程计、IMU惯性传感器以及光电编码器等

    移动机器人最初是通过自身携带的内部传感器基于航迹推算的方法进行定位 xff0c 后来进一步发展到通过各种外部传感器对环境特征进行观测从而计算出移动机器人相对于整个环境的位姿 目前为止 xff0c 形成了基于多传感器信息融合的定位方法 现有移
  • 路由器接口

    深刻认识到如果不好好学习计算机网络 xff0c 对于自己学习后台的知识有很大的阻碍 所以 xff0c 这段时间好好把这方面的知识加强一下 一般路由器上的接口分为三大类 xff1a 一 用于局域网的LAN接口 二 用于广域网接入 互联的WAN
  • UART、I2C、SPI接口常见面试问题总结

    UART 定义 xff1a Universal Asynchronous Receiver Transmitter 通用异步收发传输器 特点 xff1a 速率不快 可全双工 结构上一般由波特率产生器 UART发送器 UART接收器组成 xf
  • ubuntu18.04配置ORB-SLAM3(包含ROS)完整版教程

    ORB SLAM3安装教程 ORB SLAM3安装准备1 C 43 43 11 or C 43 43 0x Compiler2 Pangolin 61 61 出现的问题 61 61 3 OpenCV安装4 Eigen安装5 boost安装6
  • 视觉里程计--视觉slam7.1/相机运动估计视觉算法

    视觉里程计 本篇文章记录了少许阅读 视觉slam14讲 的阅读整理 xff0c 不是特别全面 xff0c 只是为了本次项目中特定任务搜查资料 xff0c 时间比较紧 xff0c 文章并没有全面涵盖所有知识点 日后若时间有空闲 xff0c 将
  • [授权发表]源码分析:动态分析 C 程序函数调用关系

    By Falcon of TinyLab org 2015 04 15 最初发表 xff1a 泰晓科技 聚焦嵌入式 Linux xff0c 追本溯源 xff0c 见微知著 xff01 原文链接 xff1a 源码分析 xff1a 动态分析 C
  • [授权发表]利用 qemu 模拟嵌入式系统制作全过程

    利用qemu模拟嵌入式系统制作全过程 by Pingbo Wen of TinyLab org 2013 08 31 最初发表 xff1a 泰晓科技 聚焦嵌入式 Linux xff0c 追本溯源 xff0c 见微知著 xff01 原文链接
  • [授权发表]Docker 快速上手:用 Docker + GitBook 写书

    By Falcon of TinyLab org 泰晓沙龙第二期 64 2015 04 26 最初发表 xff1a 泰晓科技 聚焦嵌入式 Linux xff0c 追本溯源 xff0c 见微知著 xff01 原文链接 xff1a Docker
  • 2020年最新 C# .net 面试题,月薪20K+中高级/架构师必看(一)

    笔者近几年前前后后面试了50 43 公司左右 xff0c 怎么讲呢 xff0c 每个面试官的风格都不一样 xff0c 要问的问题也不尽相同 但是面试是需要技巧的 xff0c 提前准备工作以及如何把简历写得让人眼前一亮是很有必要的 xff0c
  • [授权发表]Android加载不同DPI资源与内存消耗之间的关系

    by Will of TinyLab org 2015 04 21 最初发表 xff1a 泰晓科技 聚焦嵌入式 Linux xff0c 追本溯源 xff0c 见微知著 xff01 原文链接 xff1a Android加载不同DPI资源与内存