Android加载drawable中图片后自动缩放的原理

简介: Android加载drawable中图片后自动缩放的原理

Android加载drawable中图片后自动缩放的原理

日常开发中我们少不了要根据设计图绘制UI,一般而言设计师给的都是设计图都是750*1334的,给的切图也一般是2x3x图。

简单起见,我们只将对应的2x图标放到res/drawable-xhdpi目录下即可。

当然了,对于要求高的图标,我们需要添加对应的多套图,分别放置到drawable-mdpidrawable-hdpidrawable-xhdpidrawable-xxhdpidrawable-xxxhdpi、目录下。

ps: `drawable-ldpi`太低了,可以忽略掉。

图标的容器我们一般使用ImageView,代码如下:

<ImageView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:src="@drawable/empty_icon" />

实际开发中我们直接给对应的src属性设置对应的图片资源id即可,不同机型上的适配工作,Android系统会自动帮我们完成。

系统会自动根据当前机型的dpi,到对应的drawable目录下获取图片,接着再对图片进行适当的缩放操作,最后完成显示。

关于dpi和drawable是如何匹配的,请移步郭霖大神的Android drawable微技巧,你所不知道的drawable的那些细节

这篇文章中有一个很重要的dpi匹配的表,如下:

这里我想讲下Android在获取到对应的图片后,是如何进行对应的缩放操作的。

实例讲解

我们先从一个例子讲起。

1、图片准备

我们先准备对应的五张图片 empty_icon ,他们的分辨率分别是

mdpi        106*106
hdpi        159*159``
xhdpi        213*213
xxxhdpi     319*319
xxxhdpi        426*426

我们看下官网的说明,如下图:

这说明我们的图片是符合比例规则的,接着将他们放入到对应的drawable目录下。

ps:这几张图片来自于bravh这个库,如有侵权,请联系我,侵删。

2、展示图片信息

接着我们将这个图片设置给ImageView对象,代码如下:

<ImageView
    android:id="@+id/iv"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:layout_gravity="center_horizontal"
    android:src="@drawable/empty_icon" />

然后获取当前图片的实际宽高,代码如下:

iv.post {
    var sb = StringBuffer("图片实际宽高:\n")
    sb.append("width: " + iv.width)
    sb.append("\n")
    sb.append("height: " + iv.height)

    tvIvInfo.text = sb.toString()
}

接着运行代码,我们看下现象,效果图如下:这个测试机的dpi为440。

咦,这里图片的尺寸怎么不对呢?drawable-xxhdpi文件夹下的图片尺寸命名是319啊,这里为何不一致,而且我们也没有这个尺寸的图片啊,为什么呢?

莫慌,我们这次换一个xxhdpi的模拟器试下,运行效果如下图:

这次效果正确了吧,可是这个缩放原理是什么呢?

3、图片缩放原理解析

我们都知道Android会为我们自动缩放drawable中的图片,可是到底缩放了多少,什么情况下缩放呢?

我们先看下dpi和drawable的对应关系:需要说明的是,这里的范围都是不包含头包含尾,可以用(a,b]来表示。

我们第一次运行的测试机的dpi为440,第二次的为480,他两对应的dpi等级均是xxhdpi,所以它们对应的drawable目录为drawable-xxhdpi,我们这里对应的图片的尺寸应该是319*319

可是为啥两个手机上显示的尺寸不一样呢?一个是292*292,一个是319*319。有什么规律呢?

眼尖的同学应该已经看出来了,实际的图片尺寸缩放比例实际dpi对应dpi等级相关,规律如下:

对第一个实例来说,实际dpi这里对应的分别是440,对应的dpi通过上面的表格可以得出为480,对应drawable图片尺寸为319px,实际显示尺寸为我们获取到的292.

上述公式转换下,如下所示:计算实际显示尺寸的值。

针对某个图片来说,实际dpi对应的dpi等级对应drawable下图片尺寸这三个值都是已知的,根据这三个值我们就可以获取到实际显示的尺寸。

4、结论验证

接下来我们验证下我们这个结论。我们先写一个工具类,根据实际dpi获取对应的dpi等级,代码如下所示:

/**
 * 根据实际dpi获取对应的dpi等级。
 * 登记表参见 https://blog.csdn.net/guolin_blog/article/details/50727753
 * @param srcDpi
 * @return
 */
public static int getTargetDpi(int srcDpi) {
    int targetDpi = 0;

    if (srcDpi <= 120) {// ldpi
        targetDpi = 120;
    } else if (srcDpi <= 160) {// mdpi
        targetDpi = 160;
    } else if (srcDpi <= 240) {// hdpi
        targetDpi = 240;
    } else if (srcDpi <= 320) {// xhdpi
        targetDpi = 320;
    } else if (srcDpi <= 480) {// xxhdpi
        targetDpi = 480;
    } else if (srcDpi <= 640) {// xxxhdpi
        targetDpi = 640;
    }

    return targetDpi;
}

接着我们再写一个方法,根据对应的dpi等级,可以获取 R.drawable.empty_icon 的实际尺寸:

/**
 * 根据对应的dpi等级,获取 R.drawable.empty_icon 对应的实际尺寸
 */
fun getDrawableSize(targetDpi: Int): Int {
    var drawableSize = 0;
    when (targetDpi) {
        120, 160 -> drawableSize = 106
        240 -> drawableSize = 159
        320 -> drawableSize = 213
        480 -> drawableSize = 319
        640 -> drawableSize = 426
    }
    return drawableSize
}

接下来就是核心代码了,如下所示:

// 获取当前手机实际的dpi
var srcDpi = this.resources.displayMetrics.densityDpi

// 获取实际dpi对应的dpi等级
var targetDpi = ScreenUtils.getTargetDpi(srcDpi)

// 根据dpi等级,获取 R.drawable.empty_icon 实际尺寸
var drawableSize = getDrawableSize(targetDpi)

// 根据实际尺寸计算缩放后的尺寸
var resultSize = drawableSize * 1.0f / targetDpi * srcDpi

tvExpect.text = String.format("期望结果:%.2f", resultSize)

我们先在刚才的440dpi的手机上验证下,效果如下:

可以看出,我们计算出的期望值四舍五入后就是实际显示的值。

接下来看下mdpi的手机上的显示效果:

接下来看下hdpi的手机上的显示效果:

接下来看下xhdpi的手机上的显示效果:

接下来看下420hdpi的手机上的显示效果:

接下来看下440hdpi的手机上的显示效果:

接下来看下xxhdpi的手机上的显示效果:

接下来看下560hdpi的手机上的显示效果:

由于Android Studio自带的模拟器中没有xxxhdpi等级的模拟器,所以这里就只能使用560dpi近似展示了,他两是一个dpi等级的。

总结

通过上述分析,我们以后可以放心的使用Android自带的drawable文件进行图标的适配了,只要我们正确的提供了多套图,Android系统会自动为我们加载合适的图片,再进行适当的缩放。

之前不是很懂这个原理的时候,我都是通过给ImageView设置高度来适配的,代码如下:

<ImageView
    android:id="@+id/iv"
    android:layout_width="106dp"
    android:layout_height="106dp"
    android:layout_gravity="center_horizontal"
    android:background="@drawable/empty_icon" />

这次搞清楚原理之后,这种比较苟的方式就可以废弃了,直接使用Android提供的默认方式即可,这样ImageView的书写方式就极其简单了:

<ImageView
    android:id="@+id/iv"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:layout_gravity="center_horizontal"
    android:src="@drawable/empty_icon" />

github项目地址

具体demo地址位于:
https://github.com/tinyvampirepudge/Android_Base_Demo

具体页面:https://github.com/tinyvampirepudge/Android_Base_Demo/blob/master/app/src/main/java/com/tiny/demo/firstlinecode/drawable/DrawableTest1Activity.kt

打开方式如下:

参考

Android drawable微技巧,你所不知道的drawable的那些细节

https://developer.android.com/guide/topics/resources/providing-resources.html?hl=zh-cn#AlternativeResources

相关文章
|
4月前
|
安全 Android开发 Kotlin
Android经典实战之SurfaceView原理和实践
本文介绍了 `SurfaceView` 这一强大的 UI 组件,尤其适合高性能绘制任务,如视频播放和游戏。文章详细讲解了 `SurfaceView` 的原理、与 `Surface` 类的关系及其实现示例,并强调了使用时需注意的线程安全、生命周期管理和性能优化等问题。
206 8
|
2月前
|
缓存 Java 数据库
Android的ANR原理
【10月更文挑战第18天】了解 ANR 的原理对于开发高质量的 Android 应用至关重要。通过合理的设计和优化,可以有效避免 ANR 的发生,提升应用的性能和用户体验。
131 56
|
2月前
|
XML 前端开发 Android开发
Android:UI:Drawable:View/ImageView与Drawable
通过本文的介绍,我们详细探讨了Android中Drawable、View和ImageView的使用方法及其相互关系。Drawable作为图像和图形的抽象表示,提供了丰富的子类和自定义能力,使得开发者能够灵活地实现各种UI效果。View和ImageView则通过使用Drawable实现了各种图像和图形的显示需求。希望本文能为您在Android开发中使用Drawable提供有价值的参考和指导。
49 2
|
3月前
|
Android开发 UED
Android 中加载 Gif 动画
【10月更文挑战第20天】加载 Gif 动画是 Android 开发中的一项重要技能。通过使用第三方库或自定义实现,可以方便地在应用中展示生动的 Gif 动画。在实际应用中,需要根据具体情况进行合理选择和优化,以确保用户体验和性能的平衡。可以通过不断的实践和探索,进一步掌握在 Android 中加载 Gif 动画的技巧和方法,为开发高质量的 Android 应用提供支持。
|
4月前
|
存储 缓存 编解码
Android经典面试题之图片Bitmap怎么做优化
本文介绍了图片相关的内存优化方法,包括分辨率适配、图片压缩与缓存。文中详细讲解了如何根据不同分辨率放置图片资源,避免图片拉伸变形;并通过示例代码展示了使用`BitmapFactory.Options`进行图片压缩的具体步骤。此外,还介绍了Glide等第三方库如何利用LRU算法实现高效图片缓存。
83 20
Android经典面试题之图片Bitmap怎么做优化
|
3月前
|
XML 前端开发 Android开发
Android View的绘制流程和原理详细解说
Android View的绘制流程和原理详细解说
59 3
|
4月前
|
ARouter 测试技术 API
Android经典面试题之组件化原理、优缺点、实现方法?
本文介绍了组件化在Android开发中的应用,详细阐述了其原理、优缺点及实现方式,包括模块化、接口编程、依赖注入、路由机制等内容,并提供了具体代码示例。
62 2
|
4月前
|
编解码 前端开发 Android开发
Android经典实战之TextureView原理和高级用法
本文介绍了 `TextureView` 的原理和特点,包括其硬件加速渲染的优势及与其他视图叠加使用的灵活性,并提供了视频播放和自定义绘制的示例代码。通过合理管理生命周期和资源,`TextureView` 可实现高效流畅的图形和视频渲染。
329 12
|
3月前
|
Java 调度 Android开发
Android面试题之Kotlin中async 和 await实现并发的原理和面试总结
本文首发于公众号“AntDream”,详细解析了Kotlin协程中`async`与`await`的原理及其非阻塞特性,并提供了相关面试题及答案。协程作为轻量级线程,由Kotlin运行时库管理,`async`用于启动协程并返回`Deferred`对象,`await`则用于等待该对象完成并获取结果。文章还探讨了协程与传统线程的区别,并展示了如何取消协程任务及正确释放资源。
56 0
|
5月前
|
数据处理 开发工具 数据安全/隐私保护
Android平台RTMP推送|轻量级RTSP服务|GB28181接入之文字、png图片水印的精进之路
本文探讨了Android平台上推流模块中添加文字与PNG水印的技术演进。自2015年起,为了满足应急指挥及安防领域的需求,逐步发展出三代水印技术:第一代为静态文字与图像水印;第二代实现了动态更新水印内容的能力,例如实时位置与时间信息;至第三代,则优化了数据传输效率,直接使用Bitmap对象传递水印数据至JNI层,减少了内存拷贝次数。这些迭代不仅提升了用户体验和技术效率,也体现了开发者追求极致与不断创新的精神。