谈到Jetpack,大家都以为是一堆框架,事实上它的内容要大的多。本文以大家熟知的Preference组件为切入点,逐步探究它的前世今生。
Preference
作为设置画面的标准实现,大家都不陌生。这个组件跟随Android系统一同诞生,之后便不断地变更。先是Support库
中出现了独立版本,接着整合到了AndroidX
中,最后在Android 10的时候完全废弃了SDK版本。
Preference
组件的API设计得非常简单、清晰。
类 | 作用 |
---|---|
PreferenceActivity | 提供了Preference布局设置和查找的ListActivity |
PreferenceFragment | 展示Preference布局的专属Fragment |
Preference | 所有Preference组件的基类,预设了Preference处理的基本API |
PreferenceGroup | 扩展自Preference,用以嵌套Preference组件并内置List进行管理 |
PreferenceScreen | 扩展自PreferenceGroup,嵌套Preference组件的根布局,内部将管理列表View和对应的Adapter |
PreferenceCategory | 扩展自PreferenceGroup,展示设置条目分组的小标题,不可点击、不可获得焦点 |
SwitchPreference | 内置了Switch控件的Preference组件,类似的扩展组件还有ListPreference、EditTextPreference等 |
… |
Preference组件是Android 1.0发布就引入的元老级组件,那会RecyclerView
还未推出,自然采用经典的ListView
构建整个设置列表。
使用起来非常简单,跟普通视图的写法并无二致。
<PreferenceScreen android:title="@string/my_preference_settings">
<PreferenceCategory
android:title="@string/my_preference_general" >
<Preference
android:fragment="com.android.settings.applications.ManageApplications"
android:key="app"
android:title="@string/my_preference_general_apps" />
PreferenceCategory>
...
PreferenceScreen>
public class SettingsActivity extends PreferenceActivity {
public void onCreate(Bundle bundle) {
super.onCreate(bundle);
addPreferencesFromResource(R.xml.my_preference_layout);
}
}
原理也不复杂:
ListView的性能欠佳,不再适应复杂的设置画面,尤其是内容众多的系统设置App。
Support库
是为新API提供向后兼容性的支持库,包含大量应用组件、视图、Material Design
等功能类。重新改写的Preference
组件也包含其中。
依据兼容API版本的不同,Support库的分支众多且凌乱,使用起来也愈发繁琐和呆板。
Preference组件的变更首次出现在Support库的V7包,主要是将SDK版本的Preference组件拷贝过来进行了重写。
对外的API只是微调,区别大体集中在内部的实现细节上:
onBindViewHolder()
来向RecyclerView提供条目的视图PreferenceFragmentCompat
。使用的话需导入额外依赖:
implementation 'com.android.support:preference-v7:28.0.0'
另外要注意的是Fragment里加载布局的API由addPreferencesFromResource()改为setPreferencesFromResource()
。由于API只是微调,其他使用起来几乎没有变化。
public static class PrefsFragment extends PreferenceFragmentCompat {
@Override
public void onCreatePreferences(Bundle savedInstanceState, String rootKey) {
setPreferencesFromResource(R.xml.preferences, rootKey);
}
}
第二次变更发生在V14包,区别只是将命名里的Compat
字样去掉了,弱化了和SDK版本的API差异。
比如:
导入只需要细微调整即可:
implementation 'com.android.support:preference-v14:28.0.0'
随着Android系统逐渐流行到TV等大屏设备,Google推出了Leanback
导航模式,并引入到了V17中。Preference组件也针对Leanback模式进行了跟进,新增了一系列新组件。
新的类 | 作用 |
---|---|
BaseLeanbackPreferenceFragment | 类似瑞士军刀风格的PreferenceFragment抽象基类,内部集成了VerticalGridView控件 |
LeanbackPreferenceFragment | 外层包裹了标题的BaseLeanbackPreferenceFragment子类 |
LeanbackSettingsFragment | 根布局为LeanbackSettingsRootView的Fragment组件,主要和LeanbackPreferenceFragment配合使用 |
LeanbackPreferenceDialogFragment | 带DialogPreference的Leanback Fragment组件 |
LeanbackListPreferenceDialogFragment | 带ListPreference的LeanbackPreferenceDialogFragment组件 |
比如Support库各分支下Preference组件在AndroidX下的对应关系:
Support库 | AndroidX |
---|---|
com.android.support:preference-v7 | androidx.preference:preference |
com.android.support:preference-v14 | androidx.legacy:legacy-preference-v14 |
com.android.support:preference-leanback-v17 | androidx.leanback:leanback-preference |
使用也很方便,只需指定对应的包名和版本即可:
def preference_version = "1.1.1"
implementation "androidx.preference:preference:$preference_version"
AndroidX和原有Support库的API对应关系,可以到官方的映射表里进行查询:
包的关系:https://developer.android.com/jetpack/androidx/migrate/artifact-mappings
类的映射关系:https://developer.android.google.cn/jetpack/androidx/migrate/class-mappings
将最核心的Preference类进行对比,可以发现:除了格式、书写风格的差异以外,代码逻辑几乎完全一致。
再比如AndroidX里提供的PreferenceFragment类,其实现和Support库的版本几乎是一样的。
AndroidX replaces the original support library APIs with packages in the
androidx
namespace. Only the package and Maven artifact names changed; class, method, and field names did not change.
像官方描述的那样,AndroidX是针对Support库的整合和替代,区别仅仅体现在仓库的地址和包名。正因为此,AndroidX拥有清晰统一的版本管理,开发者能便捷和灵活地使用。
之前,Preference组件等新API分散在Support库的各个分支包里,源文件也会集成到AOSP源码,ROM厂商可以修改。
比如V14包的Preference组件在AOSP源码的对应位置如下。
/frameworks/support/v14/preference/src/android/support/v14/preference/
Android 9开始整合到了AndroidX里,但为了过渡,源文件在AOSP源码里仍然保留。也就是我们仍然可以修改其源码。
/frameworks/support/preference/src/main/java/androidx/preference/
Android 10开始全面转向AndroidX,彻底废弃Support库的使用。AOSP源码里也不再集成源文件,只提供了对应的AAR包,这也使得ROM厂商更改实现变得困难,需要额外留意。
/prebuilts/sdk/current/androidx/m2repository/androidx/preference/
为了简化向后兼容的开发工作,将Support库全面迁移至AndroidX极为必要,设置如下的Gradle 插件标志即可。
android.useAndroidX
:Android 插件会使用对应的 AndroidX 替代Support库android.enableJetifier
:Android 插件会通过重写其二进制文件来自动迁移现有的第三方库,以使用 AndroidX 依赖项当然在AndroidStudio菜单里也可以手动地迁移至AndroidX:Menu
→ Refactor
→ Migrate to AndroidX
。
更详细的迁移细节可以参考如下这篇文章:
https://www.jianshu.com/p/41de8689615d
依照官方提供的AndroidX构成列表,我概括并制作了一张AndroidX的构成图。
可以看到,实际上AndroidX在集成了Support库的以外,还涵盖了众多知名的Jetpack框架,这些框架实际上来源于2017年发布的Android Architecture Components(AAC)。
Android App开发有很多痛点,包括Activity/Fragment生命周期的管理较为呆板,线程间数据传递的复杂,SQLite封装的繁琐等等。为了改善这些状况并对App架构进行指导,Google IO 2017上发布了Android Architecture Components
,简称AAC
。
它包含了几个较为经典的框架:
Lifecycle
LiveData
ViewModel
Room
Paging
、Navigation
和WorkManager
同时Google还给Android开发者展示了推荐的应用架构,随着Jetpack家族的日益壮大,先在看来这个架构图略显简单。
AAC库在完善的过程中,和Support库一起,也逐步往AndroidX中迁移,并孕育出一个更大更强的概念Jetpack。
短短一年后,Android Architecture Components就退出了舞台,Google IO 2018上发布了全新的Jetpack开发套件。
`Jetpack`的官方构成图可以看出来:核心的Architecture
模块涵盖了熟知的框架,前身就是去年发布的AAC库
以及从Support
库整合过来的包,比如Preference
、Framgent
、AppCompat
等
除此之外,还包括KTX
和Test
工具包等
Android Jetpack is a set of libraries, tools and architectural guidance to help make it quick and easy to build great Android apps. It provides common infrastructure code so you can focus on what makes your app unique.
所以说,将Jetpack理解为一系列框架不够准确。实际上它是包含了框架、KTX、开发工具和开发向导的开发套件,期望在多个层面提升与Android开发的效率。
Jetpack开发套件的源码管理在AndroidX内,包括之前的Support库,还有后来吸收的AAC库等等。简要绘制了一下Jetpack的演变图。(画着画着,竟画成了Android机器人的形象,哈哈)
非要总结下Jetpack和AndroidX关系的话,像fundroid大神描述的那样比较贴切。
AndroidX是对SDK以外API的内部管理包,Jetpack则是对外宣传的开发套件。
“AndroidX”的名字也很酷啊,那为什么不直接用它来进行宣传?
个人的一些理解:
Android的分支众多、迭代太快,开发者疲于应对。Google一直在试图改变这种混乱局面,从经典的Support库
,到变革的AAC库
,再到持续火爆的Jetpack套件
。
与此同时,随着Android系统愈加完善,SDK也趋于稳定,一年一度的OSV终将是小修小补。但行业的持续发展必将催生层出不穷的新理念、新技术。Google自然不会停下脚步,它将以更高频次、更大范围的动作去变革和应对,而这多将聚焦在SDK以外的领域,比如Jetpack、MAD等。
MAD
,全称Modern Android Development
,是Google针对Android平台的全新开发理念。它站在比Jetpack更高的视野,旨在通过语言、工具、发行格式、框架等多个层面去指导新型的Android开发。
在Jetpack套件以外MAD还囊括了诸多内容,包括:
Android Studio
Kotlin
开发语言Android App Bundle
发行格式Compose
工具包可以说,MAD
是每个Android开发者都应了解和掌握的重要技术,后续我将解读这个全新的开发理念。
AndroidX的版本说明
Support库的说明
Jetpack的组成
基于Android Architecture Components的应用架构指南
Jetpack与AndroidX的关系
Android 12上焕然一新的小组件:美观、便捷和实用
Android 12上全新的应用启动画面,还不适配一下?
全面复盘Android开发者容易忽视的Backup功能