对于Android启动应用时候黑白屏也是困扰了好久,现在终于得到了解决。
首先先感谢:带你重新认识:Android Splash页秒开 Activity白屏 Activity黑屏 的详细分析。
然后也感谢:Android 必知必会 - Android Splash 页秒开之细节处理 的补充。
#问题的产生原因呢:
绘制整个窗口需要按顺序执行以下几个步骤:
1. 绘制背景。
2. 绘制View本身的内容。
3. 绘制子View。
4. 绘制修饰内容(例如滚动条)。
这里是主要的四步,还有些其他对于今天内容不太重要省去没写。
我们正常开发中会在Activity
的onCreate()
方法中调用setContentView(View)
设置该Activity
的显示布局,那么问题就来了,既然我们设置了布局,为什么启动的时候还会白屏或者黑屏而不是显示我set
的布局呢?下面就带领大家一起来剖析一下原因。
当打开一个Activity
时,如果这个Activity
所属Application
还没有在运行,系统会为这个Activity
的创建一个进程(每开启一个进程都会有一个Application
,所以Application
的onCreate()
可能会被调用多次),但进程的创建与初始化都需要时间,在这个动作完成之前,如果初始化的时间过长,屏幕上可能没有任何动静,用户会以为没有点到按钮。所以既不能停在原来的地方又没到显示新的界面,怎么办呢?这就有了StartingWindow
(也称之为PreviewWindow
)的出现,这样看起来就像Activity
已经启动起来了,只是数据内容还没有初始化好。
StartingWindow
一般出现在应用程序进程创建并初始化成功前,所以它是个临时窗口,对应的WindowType
是TYPE_APPLICATION_STARTING
。目的是告诉用户,系统已经接受到操作,正在响应,在程序初始化完成后实现目的UI,同时移除这个窗口。
这个StartingWindow
就是我们要讨论的白屏和黑屏的“元凶”,一般情况下我们会对Application
和Activity
设置Theme
,系统会根据设置的Theme
初始化StartingWindow
。Window
布局的顶层是DecorView
,StartingWindow
显示一个空DecorView
,但是会给这个DecorView
应用这个Activity
指定的Theme
,如果这个Activity
没有指定Theme
就用Application
的(Application
系统要求必须设置Theme
)。
在Theme
中可以指定窗口的背景,Activity
的ICON
,APP整体文字颜色等,如果说没有指定任何属性,就会用默认的属性,也就是上文中提到的空DecorView
,所以我们的白屏和黑屏和空DecorView
息息相关,我们给APP设置的Style就决定了是白屏还是黑屏。
1、如果选择了Black
的系列的主题那么Activity
跳转的时候就是黑屏:
2、如果选择了Light
的系列的主题那么Activity
跳转的时候就是白屏:
#解决方法
两篇博客里面都给出了解决办法大体是差不多的(个人更倾向于下面这一种);##延迟启动和添加淡出的过场动画
<span style="color: rgb(102, 102, 102); font-family: -apple-system, 'PingFang SC', 'Hiragino Sans GB', Arial, 'Microsoft YaHei', 'Helvetica Neue', sans-serif;font-size:14px; line-height: 27.2px; text-align: justify;">SplashActivity</span>
<span style="font-size:14px;">@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
Intent starter = new Intent(SplashActivity.this, xxxActivity.class);
startActivity(starter);
finish();
overridePendingTransition(R.anim.stand,R.anim.splash);
}
},500);
}</span>
<span style="color: rgb(102, 102, 102); font-family: -apple-system, 'PingFang SC', 'Hiragino Sans GB', Arial, 'Microsoft YaHei', 'Helvetica Neue', sans-serif; line-height: 27.2px; text-align: justify;"><span style="font-size:18px;">stand.xml</span></span><span style="font-size:14px;">
</span>
<span style="font-size:14px;"><?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:interpolator="@android:anim/accelerate_interpolator">
<translate
android:duration="200"
android:fromXDelta="0%p"
android:toXDelta="0%p"
/>
</set></span>
<span style="color: rgb(102, 102, 102); font-family: -apple-system, 'PingFang SC', 'Hiragino Sans GB', Arial, 'Microsoft YaHei', 'Helvetica Neue', sans-serif; line-height: 27.2px; text-align: justify;"><span style="font-size:18px;">splash.xml</span></span><span style="font-size:14px;">
</span>
<span style="font-size:14px;"><?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android">
<alpha
android:fromAlpha="1.0"
android:toAlpha="0.0"
android:duration="200"
/>
</set></span>
在此也只是做一个记录,也希望更多的Android开发者看见和学习。