绘制整个App窗口需要按顺序执行以下几个步骤:
1. 绘制背景。
2. 绘制View本身的内容。
3. 绘制子View。
4. 绘制修饰内容(例如滚动条)。
闪屏原因剖析StartingWindow(Preview Window)
我们正常开发中会在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就决定了是白屏还是黑屏。
如果选择了Black的系列的主题那么Activity跳转的时候就是黑屏:
@android:style/Theme.Black"
如果选择了Light的系列的主题那么Activity跳转的时候就是白屏:
@android:style/Theme.Light"
解决方法
弄一个颜色或者图片来代替白屏和黑屏:
1.首先在res/drawable下新建一个layer-list,名字随便取,比如splash.xml,代码如下:
<?xml version="1.0" encoding="utf-8"?>
<layer-list xmlns:android="http://schemas.android.com/apk/res/android">
<item
android:drawable="@color/gray"/>
<item>
<bitmap
android:gravity="center" //图片位置居中
android:src="@mipmap/ic_launcher"/> //图片自己选择
</item>
</layer-list>
2.在styles文件中新建主题如:SplashTheme,代码如下:
<style name="SplashTheme" parent="AppBaseTheme">
<!-- 调用上面写的layer-list(也可以直接调用一张图片),并让它全屏-->
<item name="android:windowBackground">@drawable/splash</item>
<item name="android:windowFullscreen">true</item>
</style>
3.在AndroidManifest.xml中定义SplashActivity的theme为SplashTheme:
<activity android:name=".SplashActivity"
android:theme="@style/SplashTheme">
<intent-filter>
<action android:name="android.intent.action.MAIN"/>
<category android:name="android.intent.category.LAUNCHER"/>
</intent-filter>
</activity>
4.SplashActivity的实现,在onCreate()启动你的MainActivity即可,其他什么都别干(不要调用setContentView()方法,加快速度):
public class SplashActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
startActivity(new Intent(this, MainActivity.class));
finish();
}
}
5.如果你需要初始化一些数据,可以跳转到其他 Activity,然后再进入MainActivity
转载声明
本文转自严振杰的博客,特此感谢!