那天遇到一个资源目录层级的问题,索性重新整理记录一下,希望能帮到如吾往昔之少年的你们,哈哈哈哈哈哈...
一脸茫然,越写越多,时间成本属实有点大,就当一起来基础扫盲吧
-
- res
- assets(&res区别)
- 兴趣衍生
-
- [res 和 asset 都是存储的静态资源吗?](#res 和 asset 都是存储的静态资源吗?)
- [res下 raw 和 assets 有何区别?](#res下 raw 和 assets 有何区别?)
- drawable-v24、mipmap-anydpi-v26、mipmap-anydpi-v33
新建了一个Demo,以下为初始目录,那么就从这里开始讲吧
Android
.Project(更全面、清晰)
res
主要存放静态资源,不过存储的都是资源文件,同时资源分类较多,该类文件在打包时会直接通过 aapt
(资源文件打包工具)打包res资源文件,然后映射到Android工程的R文件中,生成 R.资源文件对应的ID,通常都可通过 R + 资源类型 + 资源具体名称(R.drawable.ic_launcher_background)
访问到资源文件(注意:raw下的文件会被原封不动的打包到apk中)
常用与否,仅取决于个人定义
常用
drawable (res/drawable)
:存放图片资源,如图标、背景等mipmap (res/mipmap)
:存放应用的启动器图标资源,为不同屏幕密度提供不同版本的图标values (res/values)
:存放配置值,如字符串、样式、颜色、尺寸等layout (res/layout)
:存放XML布局文件,定义用户界面的结构(常见于Activity、Fragment、Dialog等组件的视图xml )dimens (res/dimens)
:存放尺寸资源文件,定义间距、字体大小等color (res/color)
:存放颜色资源文件menu (res/menu)
:存放菜单资源文件,定义应用的菜单项 - Android进阶之路 - BottomNavigationView的使用与问题处理方案xml (res/xml)
:存放XML文件,如网络配置、偏好设置等 - Android 9.0 兼容适配font (res/font)
:存放自定义字体文件(也支持放在 assets 目录) - Android进阶之路 - 深入浅出字体、字体库
关于动画一般放置于 anim
下,其他的更细分一些,好像尚未使用过
anim (res/anim)
:存放动画资源文件,定义视图的动画效果animator (res/animator)
:存放属性动画XML文件transition (res/transition)
:存放场景转换动画资源
raw (res/raw)
:存放原始文件,如音频、视频文件等(有些特殊,单独讲讲)
raw
与 asset
有些类似,它们存储的文件范围一般为音频、视频等,而且均在打包时不会被压缩,简单来看它们存储的位置不同,当然区别点还挺多,我在兴趣扩展中有详细介绍俩者之间的具体区别
不常用
layout-<version> (res/layout-land, res/layout-port)
:为不同布局方向或屏幕尺寸提供特定的布局
values 限制场景
values-<language> (res/values-en, res/values-fr, etc.)
:为不同语言提供特定的字符串资源(适用多语言定制,一般用于海外app)values-v21 (res/values-v21)
:为API级别21及以上版本提供特定的资源values-night (res/values-night)
:为夜间模式提供特定的资源
关于 drawable
与 mipmap
都说到了图片视频,之所以进行图片适配,主要有以下几点原因
- 一致性、适应性:保持界面的统一视觉效果,同时都足够清晰
- 性能:避免在高密度屏幕上使用过大的图像文件,从而减少内存使用和提高加载速度(相对的因为图片适配,会适当增大一些apk体积,可适当在不影响图片的效果下进行图片压缩)
drawable
存放种类比较广,兼容性较强,支持存入以下类型 - 当然还有一些其他
xml
标签,但是因为一时想不起来,或者不常用就先不总结了,若有有需要再来补充
- 支持存放 图片(JPEG、PNG、GIF、WebP 等格式) ,在
mipmap
目录未出现前,统一将图片放于drawable
下,同时此前图片适配也在该处
drawable
存放不区分密度的图片资源,这些资源将被所有密度级别的设备使用(假设未进行图片适配,那么有可能会加载这里的图片)
drawable-mdpi
中等密度,基准密度,大约为160dpi,Android的基准密度drawable-hdpi
高密度,大约240dpidrawable-xhdpi
超高密度,大约320dpidrawable-xxhdpi
超超高密度,大约480dpidrawable-xxxhdpi
超超超高密度,大约640dpi
- 支持
shape
、selector
、Layer-list 等
xml 标签
shape
支持实现圆角、矩形、圆形等效果 - Android入门之路 - shape保姆级手册selector
状态选择器,支持点击、按压等状态监听后的不同实现效果 - Android进阶之路 - selector状态选择器Layer-list
视图层,涉及层级覆盖,根据场景使用(之前做阴影效果时有接触过)- Android进阶之路 - 背景阴影、阴影背景
- 支持
animation-list
帧动画 xml 标签 - Android入门之路 - Frame帧动画
mipmap
在 mipmap
出现之前,我们一般会将图片统一放置于 drawable
目录,在其出现之后我们大多将图片放于该目录下,实现了单一职责
在前端、客户端经常需要进行适配,其中图片适配必不可少,针对于手机的不同分辨率,为了用户有更好的体验,通常需要在对应目录下放置同名不同分辨率
的图片,目前使用较多的目录主要有 mipmap-xhdpi
、mipmap-xxhdpi
mipmap-mdpi
中密度屏幕(Medium Density),大约为160dpi,Android的基准密度mipmap-hdpi
高密度屏幕(High Density),大约为240dpimipmap-xhdpi
超高密度屏幕(Extra High Density),大约为240dpimipmap-xxhdpi
超超高密度屏幕(Extra Extra High Density),大约为320dpimipmap-xxxhdpi
超超超高密度屏幕(Extra Extra Extra High Density),大约为480dpi
Tip:当系统运行时一般会根据手机分辨率加载对应目录下的图片,如果该目录没有相关图片,就会就近同名的其他目录下的图片
values
values
目录下的资源也是常用的核心资源目录之一,在其内部的资源类别也比较多
从values
与 values-night
可以看出相比其他资源目录,values
提供了常规模式和夜间模式的区别 ,接下来我们看看具体有哪些常见资源文件
常用
strings.xml
:存放应用中使用的所有字符串资源styles.xml
:定义应用的样式和主题,这些样式可以应用于应用的布局和控件(共性布局抽取,主题样式声明)colors.xml
:定义应用的颜色资源,可以用于文本、背景、控件等(颜色管理)dimens.xml
:定义应用的尺寸资源,如字体大小、间距、边距等(尺寸适配)attrs.xml
:定义自定义属性,这些属性可以用于样式和主题
不常用
arrays.xml
:定义字符串数组、整数数组等integers.xml
:存放整数值,例如版本号、动画持续时间等bools.xml
:存放布尔值,用于配置开关选项plurals.xml
:用于定义复数规则,根据数量的不同显示不同的字符串themes.xml
:定义应用的主题,可以包含样式和其他主题相关的配置 - Android10.0 特性 - 暗黑模式、深色主题ids.xml
:定义资源ID,虽然通常这些ID是在XML布局文件中定义的config.xml
:定义应用的配置选项,如屏幕方向、导航键配置等preferences.xml
:定义应用的偏好设置,用于PreferenceActivitypublic.xml
:定义公共资源,这些资源可以在应用之外被其他应用访问
color
从 values
中可以看到有一个 colors.xml
文件,那么它和 res-color
目录有何区别?
values
-colors.xml
更多的是声明某个颜色的色值res-color
下的xml
文件可以提供对应事件的色值,效果更单一(区别于drawable
下的selector
标签)
xml
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<item android:state_pressed="true" android:color="#FF4081"/>
<item android:state_focused="true" android:color="#FFA726"/>
<item android:color="#9E9E9E"/>
</selector>
drawable
- selector
标签
xml
<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<item android:state_checked="true">
<shape>
<solid android:color="@color/white" />
</shape>
</item>
<item android:state_checked="false">
<shape>
<solid android:color="#f9f9f9" />
</shape>
</item>
<item>
<shape>
<solid android:color="#f9f9f9" />
</shape>
</item>
</selector>
assets(&res区别)
首先说明一下 assets
和 res
属于同级关系,别放错位置
assets
也用于存放静态资源,只不过存储的是原始文件 ,常用于存放 文本文件、音频文件、视频文件、图像文件、HTML文件、CSS文件、JavaScript文件
等
除去对比区别特性外,额外记录俩条特性
- 不支持资源重载:当APK被安装后,assets中的文件不能被更新或重载,除非重新安装APK(未验证,按理apk更新后同文件可覆盖, 不确定重载场景的产生)
- 与NDK交互:可以与NDK代码交互,NDK可以访问assets目录下的文件(未验证,关于NDK方面经验有限,不做误导)
一般我们都会将 asset 和 res 做对比
为方便下方简介对比,简化名称
res = R
assets = A
访问方式
R :在.R文件中生成对应的资源ID,可通过 R + 资源类型 + 资源具体名称(R.drawable.ic_launcher_background·)
访问到资源文件
A :需要通过 AssetManager
来访问,不能通过资源ID直接访问
编译、压缩
R :在应用编译时会被处理,如,图片文件可以被压缩(减少APK大小),布局文件(.xml)会被转换成View对象等
A:在打包时文件不会被压缩,以原始形式在APK中存在,不经过编译处理,可以保留资源文件的原始结构和内容,便于应用程序在运行时动态地读取和使用
目录(文件)结构
R :不同类型文件被存储于不同资源目录下,如 drawable、mipmap、layout、values
等,且编译后不保留目录结构
A:可以包含文件夹和子文件夹,文件的目录结构在APK中得以保留
适用场景
R :适用于存储应用的界面资源,如图片、布局、样式、颜色定义等
A:适用于存储不希望被编译或需要保持原始目录结构的文件,如配置文件、原始数据文件、大型媒体文件等
访问效率
R :由于资源在编译时被处理,访问速度通常较快
A :访问速度可能较慢,因为需要从APK中通过I/O
操作来读取原始文件
资源可见性
R :资源默认是私有的,但可以通过特定的方式(如Content Provider
)被其他应用访问
A :资源默认是私有的,只能通过 AssetManager
访问
资源管理
R :资源管理由系统自动完成,如根据屏幕密度选择合适的图片资源
A:需要手动管理,如遍历文件夹和读取文件
资源命名
R :资源文件名不区分大小写
A:文件名区分大小写(大小写敏感)
兴趣衍生
res 和 asset 都是存储的静态资源吗?
从一定层面可以说 res 和 asset 都是存储的静态资源,但是不同点在于静态资源也有所区分, 如果你有从上自下看过的话,可以看出俩者区别
文件类型 - 它们都属于静态资源文件,俩种文件类型区别上面有也说明,可以从访问方式、存储资源类型等方面来综合考虑
res
:存放的是资源文件assets
: 存放的是原始文件
res下 raw 和 assets 有何区别?
我很诧异于res下既然都已经有了raw用于存储原始文件,为何还要再搞一个assets?
后面AI搜了搜,发现还是有一些区别的,除了上面在assets提到的区别之外,还有以下区别
适用场景
res/raw
:适合存放小到中等大小的文件
,如音频片段、视频片段、JSON配置文件等。assets
:适合存放较大的文件或需要保持原始目录结构的文件
,如整个文件夹的文档、大型数据库文件等
文件操作
res/raw
:文件操作通常通过openRawResource()
方法进行assets
:文件操作通过AssetManager的open()
方法进行
关于最终是选择使用 res/raw
还是 assets
取决于你的具体需求,如文件大小、访问速度、是否需要保持目录结构等因素
drawable-v24、mipmap-anydpi-v26、mipmap-anydpi-v33
在上方的时候有看到这里的目录吗?一起来扫个盲...
Tip: 假设 v24是限制7.0版本,那么v25、v30是不是都可以做限制?
(未尝试过,大多app为保持高兼容性,这种限制应该有一些场景局限性,具体场景具体定义吧)
drawable
目录上方已经具体解释过了,不区分 Android 版本均可正常使用drawable-v24
目录中v24
指的是API级别24,对应于Android 7.0(Nougat),也就是仅支持7.0以上的机型加载对应资源,低于这个版本的都不会访问该目录下资源
以mipmap-anydpi-v33
为例,适用于API级别33及以上的机型,对于API级别低于33的机型,系统会回退到使用传统的mipmap
目录下的图标资源(v26同理,针对于API26
机型)
mipmap
代表图片anydpi
代表任意密度v33
代表版本兼容
具体看一下 ic_launcher.xml
(标签较全)
标签解析
xml
<?xml version="1.0" encoding="utf-8"?>
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
<background android:drawable="@drawable/ic_launcher_background" />
<foreground android:drawable="@drawable/ic_launcher_foreground" />
<monochrome android:drawable="@drawable/ic_launcher_foreground" />
</adaptive-icon>
adaptive-icon
:根元素,用于定义自适应图标;它包含一个命名空间声明xmlns:android
,指定了Android资源的XML命名空间background
:定义了图标的背景部分;android:drawable
属性指定了一个drawable
资源IDforeground
:定义了图标的前景部分,通常是图标的主要视觉元素;android:drawable
属性指向一个drawable
资源monochrome
:定义了一个单色版本的前景drawable
,用于在某些情况下(如深色背景上)提供更好的可见性(默认为foreground
资源)
资源指向