Android开发笔记——如何正确实现App启动页
启动页在当前的Android APP中有着广泛的使用,在Material Design规范出来之前,Android官方是不推荐使用启动页的,理由是让用户在启动APP后干等是一种浪费时间的行为。Material Design 则制定了LaunchScreen 规范,用于在APP启动的时候合理的利用从启动到实际渲染出主界面之间这段时间,显示一个LauncherScreen用来展示一些品牌元素等信息,避免显示一个枯燥的空白界面。
APP启动过程
APP的启动过程可以参考这里。这里主要简单介绍下APP冷启动的过程。
APP冷启动开始时,系统会做以下三个任务:
- 加载并启动app
- 展示一个空白Window
- 创建app进程
在上面的第二个步骤中,WindowManager 使用Activity主题中定义的背景和状态栏颜色的属性来绘制一个临时的展示UI。
当APP进程创建后,APP进程会做以下几件事:
- 创建app对象
- 创建app的主进程
- 创建MainActivity
- inflating views
- 执行Layout过程
- 执行界面绘制
当首次绘制完成后,系统会将之前展示的空白window移除,并展示创建好的MainActivity。在某些大型APP中,这个过程可能会有1-5秒。
启动页实现原理
从上面的启动分析可以知道,WindowManager是根据Activity主题中定义的背景等属性来创建一个临时preview window的。我们在Activity的主题style中设置windowBackground属性来设置这个preview window的显示内容。使用layer-list可以创建一个包含logo的drawable对象:
<layer-list xmlns:android="http://schemas.android.com/apk/res/android" android:opacity="opaque"> <item android:drawable="@android:color/white"/> <item> <bitmap android:src="@drawable/logo" android:gravity="center"/> </item> </layer-list>
创建一个自定义Theme:
<style name=“LaunchScreenTheme" parent="Theme.AppCompat.NoActionBar"> <item name="windowNoTitle">true</item> <item name="android:windowContentOverlay">@null</item> <item name="android:windowFullscreen">true</item> <item name="android:windowBackground”>@drawable/launchscreen</item> </style>
最后在Manifest中将这个主题赋值给待启动的Activity即可
<activity ... android:theme="@style/AppTheme.Launcher" />
实现方式
谷歌官方推荐的启动页实现方式是在将LauncherScreen和MainActivity合二为一。首先创建按照之前说的原理创建一个启动页Theme,并在Manifest中将其作为MainActivity的默认Theme。然后在MainActivity的onCreate方法中,在onCreate()和setContentView()之前使用setTheme方法,重新设置主界面的主题。
public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { // Make sure this is before calling super.onCreate setTheme(R.style.Theme_MyApp); super.onCreate(savedInstanceState); // ... } }
这样做的好处有以下几点:
- 不需要额外的Activity实现启动页
- 没有overdraw的问题,因为重新设置主题后,之前的绘画层也一并移除了。
当然,如果你需要根据schema或者其他方式从启动页路由到不同的界面的功能,使用单独的启动页Activity也是一个不错的选择。