【Android 内存优化】Bitmap 内存占用计算 ( Bitmap 图片内存占用分析 | Bitmap 内存占用计算 | Bitmap 不同像素密度间的转换 )

简介: 【Android 内存优化】Bitmap 内存占用计算 ( Bitmap 图片内存占用分析 | Bitmap 内存占用计算 | Bitmap 不同像素密度间的转换 )

文章目录

一、Bitmap 内存占用

二、Bitmap 内存占用计算示例

三、Bitmap 内存占用与像素密度

四、Bitmap 内存占用与像素密度示例





一、Bitmap 内存占用


在 Android 中 Bitmap 对象在内存中存储的的像素格式有两种 : ARGB_8888 和 RGB_555 ;



① ARGB_8888 像素格式 : Alpha ( 透明度 ) , Red ( 红 ) , Green ( 绿 ) , Blue ( 蓝 ) , 各占 1 11 字节 , 每个像素点占 4 字节 , 一张宽度 W WW, 高度 H HH 的图片 , 在内存中的大小是 W × H × 4 W \times H \times 4W×H×4 字节 ;


② RGB_555 像素格式 : Red ( 红 ) 占 5 55 位 , Green ( 绿 ) 占 6 66 位 , Blue ( 蓝 ) 占 5 55 位 , 每个像素点占 5 + 6 + 5 = 16 5 + 6 + 5 = 165+6+5=16 位 , 2 22 字节 , 一张宽度 W WW, 高度 H HH 的图片 , 在内存中的大小是 W × H × 2 W \times H \times 2W×H×2 字节 ;



Android 中 Bitmap 在内存中的大小与图片大小无关 , 只与像素格式和像素点个数有关 ;


内存中的大小只与分辨率有关 , 与磁盘大小无关 ;






二、Bitmap 内存占用计算示例


1. 获取 Bitmap 最小字节数 : 调用 Bitmap 对象的 getByteCount 方法 , 可以获取到 Bitmap 对象对应图像在内存中占用的最小字节数 ;


// 从资源文件中加载内存
Bitmap bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.blog);
// 打印 Bitmap 对象的宽高, 字节大小
Log.i("Bitmap", bitmap.getWidth() + " , " +
                bitmap.getHeight() + " , " +
                bitmap.getByteCount());



2. 打印结果 : 宽度 5224 像素 , 高度 2678 像素 , 内存中大小为 55959488 字节 ;


2020-06-29 20:32:12.794 9675-9675/kim.hsl.bm I/Bitmap: 5224 , 2678 , 55959488



3. Bitmap 占内存大小计算 : Android 中默认使用 ARGB_8888 像素格式 , 每个像素点占 4 44 字节 , 上图宽 5224 , 高 2678;


5224 × 2678 × 4 = 55 , 959 , 488 5224 \times 2678 \times 4 = 55,959,488

5224×2678×4=55,959,488


最终 Bitmap 在内存中的大小是 55,959,488 字节 ;






三、Bitmap 内存占用与像素密度


1 . BitmapFactory.Options 中封装了两个像素密度相关的值 :



① inDensity 像素密度值 : 表示该 Bitmap 图像的像素密度值 ;


/**
 * Bitmap 图像的像素密度 ; 
 * Bitmap.setDensity(int) 操作会导致被返回的图像会被强制设置一个像素密度值 ; 
 * 假如该设置的像素密度值 inDensity 与 目标像素密度值 inTargetDensity 不同 ,  
 * 并且 inScaled 被设置成 true , 那么该 Bitmap 就会被缩放到 inTargetDensity 对应的像素密度 ,
 * 然后再返回 ; 
 * 
 * 如果该值是 0 , 那么就默认该像素密度值就是资源文件对应的像素密度值 ;
 */
public int inDensity;



② inTargetDensity 目标像素密度值 : 表示要缩放到的目标图像像素密度值 ;


/**
 * 将要被绘制的目标像素密度值 ;
 * 该值需要结合 inScaled 值使用 , 如果同时设置了 inScaled = true , 和 inDensity 像素密度值 , 
 * 在图像返回时 , 会自动将图像按照 inDensity 向 inTargetDensity 缩放 ; 
 */
public int inTargetDensity;



如果 inDensity 小 , inTargetDensity 大 , 图像会被放大到原图像的 inTargetDensity / inDensity 倍 ;


如果 inDensity 大 , inTargetDensity 小 , 图像会被缩小到原图像的 inTargetDensity / inDensity 倍 ;




2 . 设计图片在资源文件中放置规则 :



① 设计稿分辨率 480 x 320 : 图片放在 mdpi 像素密度下 ; density 1, densityDpi 160 ;


② 设计稿分辨率 800 x 480 : 图片放在 hdpi 像素密度下 ; density 1.5, densityDpi 240;


③ 设计稿分辨率 1280 x 720 : 图片放在 xhdpi 像素密度下 ; density 2, densityDpi 320;


④ 设计稿分辨率 1920 x 1080 : 图片放在 xxhdpi 像素密度下 ; density 3, densityDpi 480;



屏幕密度 density , 屏幕像素密度 densityDpi , 关系是 density x 160 = densityDpi ;



3 . 获取当前的手机像素密度值 : 调用如下代码 , 获取当前手机屏幕的像素密度值 ;


getResources().getDisplayMetrics().densityDpi


获取的测试机的像素密度是 420 ;






四、Bitmap 内存占用与像素密度示例


1 . 不同屏幕密度资源适配 : 原图 1990 x 1020 ;


将同样大小的图片 , 分别拷贝到不同的目录 , 并命名 , 打印结果 :


代码示例 :


package kim.hsl.bm;
import androidx.appcompat.app.AppCompatActivity;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.Bundle;
import android.util.Log;
import android.widget.TextView;
public class MainActivity extends AppCompatActivity {
    static {
        System.loadLibrary("native-lib");
    }
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        TextView tv = findViewById(R.id.sample_text);
        tv.setText(stringFromJNI());
        Log.i("Bitmap", "getResources().getDisplayMetrics().densityDpi : " +
                getResources().getDisplayMetrics().densityDpi +
                " , getResources().getDisplayMetrics().density : " +
                getResources().getDisplayMetrics().density);
        // 从资源文件中加载内存
        Bitmap bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.blog);
        // 打印 Bitmap 对象的宽高, 字节大小
        Log.i("Bitmap", "blog : " + bitmap.getWidth() + " , " +
                        bitmap.getHeight() + " , " +
                        bitmap.getByteCount());
        bitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.blog_h);
        // 打印 Bitmap 对象的宽高, 字节大小
        Log.i("Bitmap", "blog_h : " + bitmap.getWidth() + " , " +
                bitmap.getHeight() + " , " +
                bitmap.getByteCount());
        bitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.blog_m);
        // 打印 Bitmap 对象的宽高, 字节大小
        Log.i("Bitmap", "blog_m : " + bitmap.getWidth() + " , " +
                bitmap.getHeight() + " , " +
                bitmap.getByteCount());
        bitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.blog_x);
        // 打印 Bitmap 对象的宽高, 字节大小
        Log.i("Bitmap", "blog_x : " + bitmap.getWidth() + " , " +
                bitmap.getHeight() + " , " +
                bitmap.getByteCount());
        bitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.blog_xx);
        // 打印 Bitmap 对象的宽高, 字节大小
        Log.i("Bitmap", "blog_xx : " + bitmap.getWidth() + " , " +
                bitmap.getHeight() + " , " +
                bitmap.getByteCount());
        bitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.blog_xxx);
        // 打印 Bitmap 对象的宽高, 字节大小
        Log.i("Bitmap", "blog_xxx : " + bitmap.getWidth() + " , " +
                bitmap.getHeight() + " , " +
                bitmap.getByteCount());
    }
    public native String stringFromJNI();
}




2 . 执行结果 :


2020-06-29 21:32:59.398 12296-12296/kim.hsl.bm I/Bitmap: getResources().getDisplayMetrics().densityDpi : 420 , getResources().getDisplayMetrics().density : 2.625
2020-06-29 21:32:59.551 12296-12296/kim.hsl.bm I/Bitmap: blog : 5224 , 2678 , 55959488
2020-06-29 21:32:59.628 12296-12296/kim.hsl.bm I/Bitmap: blog_h : 3483 , 1785 , 24868620
2020-06-29 21:32:59.775 12296-12296/kim.hsl.bm I/Bitmap: blog_m : 5224 , 2678 , 55959488
2020-06-29 21:32:59.828 12296-12296/kim.hsl.bm I/Bitmap: blog_x : 2612 , 1339 , 13989872
2020-06-29 21:32:59.864 12296-12296/kim.hsl.bm I/Bitmap: blog_xx : 1741 , 893 , 6218852
2020-06-29 21:32:59.894 12296-12296/kim.hsl.bm I/Bitmap: blog_xxx : 1306 , 669 , 3494856



3 . 结果分析 :



本测试机 : 屏幕密度 density = 2.625 , 屏幕像素密度 densityDpi = 420


原图 1990 x 1020 ;



① 图片放在 hdpi : 该像素密度对应 density = 1.5 , densityDpi = 240 ;


加 载 到 内 存 的 宽 度 = 1990 × 2.625 1.5 = 3 , 482.5 ‬ 加载到内存的宽度 = 1990 \times \dfrac{2.625}{1.5} = 3,482.5‬加载到内存的宽度=1990×

1.5

2.625


=3,482.5‬


加 载 到 内 存 的 高 度 = 1020 × 2.625 1.5 = 1785 加载到内存的高度 = 1020\times \dfrac{2.625}{1.5} = 1785加载到内存的高度=1020×

1.5

2.625


=1785



② 图片放在 mdpi : 该像素密度对应 density = 1 , densityDpi = 160;


加 载 到 内 存 的 宽 度 = 1990 × 2.625 1 = 5 , 223.75 加载到内存的宽度 = 1990 \times \dfrac{2.625}{1} = 5,223.75加载到内存的宽度=1990×

1

2.625


=5,223.75


加 载 到 内 存 的 高 度 = 1020 × 2.625 1 = 2 , 677.5 加载到内存的高度 = 1020\times \dfrac{2.625}{1} = 2,677.5加载到内存的高度=1020×

1

2.625


=2,677.5



③ 图片放在 xhdpi : 该像素密度对应 density = 2 , densityDpi = 320;


加 载 到 内 存 的 宽 度 = 1990 × 2.625 2 = 2 , 611.875 ‬ ‬ 加载到内存的宽度 = 1990 \times \dfrac{2.625}{2} = 2,611.875‬‬加载到内存的宽度=1990×

2

2.625


=2,611.875‬‬


加 载 到 内 存 的 高 度 = 1020 × 2.625 2 = 1 , 338.75 加载到内存的高度 = 1020\times \dfrac{2.625}{2} = 1,338.75加载到内存的高度=1020×

2

2.625


=1,338.75



④ 图片放在 xxhdpi : 该像素密度对应 density = 3 , densityDpi = 480;


加 载 到 内 存 的 宽 度 = 1990 × 2.625 3 = 1 , 741.25 ‬ 加载到内存的宽度 = 1990 \times \dfrac{2.625}{3} = 1,741.25‬加载到内存的宽度=1990×

3

2.625


=1,741.25‬


加 载 到 内 存 的 高 度 = 1020 × 2.625 3 = 892.5 ‬ 加载到内存的高度 = 1020\times \dfrac{2.625}{3} = 892.5‬加载到内存的高度=1020×

3

2.625


=892.5‬



这样原像素密度图片转换成目标像素密度图片后 , 就会得到日志中打印出来的值 ;


目录
打赏
0
0
0
0
39
分享
相关文章
监控和分析 JavaScript 内存使用情况
【10月更文挑战第30天】通过使用上述的浏览器开发者工具、性能分析工具和内存泄漏检测工具,可以有效地监控和分析JavaScript内存使用情况,及时发现和解决内存泄漏、过度内存消耗等问题,从而提高JavaScript应用程序的性能和稳定性。在实际开发中,可以根据具体的需求和场景选择合适的工具和方法来进行内存监控和分析。
|
4月前
|
动态内存分配与管理详解(附加笔试题分析)(上)
动态内存分配与管理详解(附加笔试题分析)
102 1
【C++核心】C++内存分区模型分析
这篇文章详细解释了C++程序执行时内存的四个区域:代码区、全局区、栈区和堆区,以及如何在这些区域中分配和释放内存。
78 2
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面,旨在通过综合策略提升程序性能,满足实际需求。
101 1
|
3月前
|
如何使用内存快照分析工具来分析Node.js应用的内存问题?
需要注意的是,不同的内存快照分析工具可能具有不同的功能和操作方式,在使用时需要根据具体工具的说明和特点进行灵活运用。
79 3
【Azure App Service】部署在App Service上的.NET应用内存消耗不能超过2GB的情况分析
x64 dotnet runtime is not installed on the app service by default. Since we had the app service running in x64, it was proxying the request to a 32 bit dotnet process which was throwing an OutOfMemoryException with requests >100MB. It worked on the IaaS servers because we had the x64 runtime install
使用 Chrome 浏览器的内存分析工具来检测 JavaScript 中的内存泄漏
【10月更文挑战第25天】利用 Chrome 浏览器的内存分析工具,可以较为准确地检测 JavaScript 中的内存泄漏问题,并帮助我们找出潜在的泄漏点,以便采取相应的解决措施。
550 9
【灵码助力Cuda算法分析】分析共享内存的矩阵乘法优化
本文介绍了如何利用通义灵码在Visual Studio 2022中对基于CUDA的共享内存矩阵乘法优化代码进行深入分析。文章从整体程序结构入手,逐步深入到线程调度、矩阵分块、循环展开等关键细节,最后通过带入具体值的方式进一步解析复杂循环逻辑,展示了通义灵码在辅助理解和优化CUDA编程中的强大功能。
Android经典面试题之图片Bitmap怎么做优化
本文介绍了图片相关的内存优化方法,包括分辨率适配、图片压缩与缓存。文中详细讲解了如何根据不同分辨率放置图片资源,避免图片拉伸变形;并通过示例代码展示了使用`BitmapFactory.Options`进行图片压缩的具体步骤。此外,还介绍了Glide等第三方库如何利用LRU算法实现高效图片缓存。
94 20
Android经典面试题之图片Bitmap怎么做优化
动态内存分配与管理详解(附加笔试题分析)(下)
动态内存分配与管理详解(附加笔试题分析)(下)
69 2

热门文章

最新文章

  • 1
    Android实战经验之Kotlin中快速实现MVI架构
    17
  • 2
    即时通讯安全篇(一):正确地理解和使用Android端加密算法
    22
  • 3
    escrcpy:【技术党必看】Android开发,Escrcpy 让你无线投屏新体验!图形界面掌控 Android,30-120fps 超流畅!🔥
    39
  • 4
    【01】噩梦终结flutter配安卓android鸿蒙harmonyOS 以及next调试环境配鸿蒙和ios真机调试环境-flutter项目安卓环境配置-gradle-agp-ndkVersion模拟器运行真机测试环境-本地环境搭建-如何快速搭建android本地运行环境-优雅草卓伊凡-很多人在这步就被难倒了
    113
  • 5
    Cellebrite UFED 4PC 7.71 (Windows) - Android 和 iOS 移动设备取证软件
    40
  • 6
    【03】仿站技术之python技术,看完学会再也不用去购买收费工具了-修改整体页面做好安卓下载发给客户-并且开始提交网站公安备案-作为APP下载落地页文娱产品一定要备案-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
    54
  • 7
    Android历史版本与APK文件结构
    147
  • 8
    【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
    46
  • 9
    【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
    39
  • 10
    APP-国内主流安卓商店-应用市场-鸿蒙商店上架之必备前提·全国公安安全信息评估报告如何申请-需要安全评估报告的资料是哪些-优雅草卓伊凡全程操作
    66