「这是我参与11月更文挑战的第16天,活动详情查看:2021最后一次更文挑战」
调整系统字体大小的之后把app从后台切换到前台,发现app crash。
关于此问题的官方文档:developer.android.google.cn/guide/topic…
一、问题原因
查阅了一些资料,发现在app中文字标注的单位为sp。当系统字体大小发生变化的时候sp的度量标准发生了变化,这个时候,系统会给app发送一个通知,告诉它配置已经变了,需要重新刷新界面。类似的操作就像是横竖屏切换,在onConfigurationChanged()方法里面进行需要的操作的配置。
横竖屏切换,app走的生命周期方法是onDestroy(),onCreate()。一般我们的应用通常配置是设定一个固定的方向让它避免横竖屏切换。通过配置configChanges属性,防止界面的销毁和重建。
android:screenOrientation="portrait"
android:configChanges="orientation|screenSize|keyboardHidden
复制代码
回到问题本身,发生问题的原理是这样的:系统字体大小发生了变化,它在app回到前台的时候发送了一个通知告诉app。app此时走的生命周期的方法是
onDestroy()
onCreate()
onResume()
复制代码
通过这样的操作进行UI界面的刷新。
重点来了:此时app是作为一种异常走的onDestroy()方法,并不是正常的按返回键进行的退出执行的onDestroy()。这个时候系统会自动在onSaveInstanceState()方法里面临时性的保存之前的界面状态或者变量信息。方便在onCreate()的时候,从bundle里面直接取出来使用。但是毕竟是异常的状态,有些实体的信息或者fragment堆栈的状态恢复的时候就有可能错乱,导致显示效果出错或者crash。
二、解决方法
android:configChanges="orientation|screenSize|keyboardHidden|**fontScale**
复制代码
- 添加fontScale属性,避免系统字体发生变化的时候,重新create界面。如果仅仅这样处理,app的在从后台切换到前台的时候,字体会随着系统的字体大小发生变化。
- 把app里面字体的单位,由sp改为dp,这样,app的字体展示就不会随着系统的字体变化而发生变化。
可以根据产品的需要自行进行配置。
三、其他问题
解决完上面的问题了,有善于思考的同学就问了,这个是解决切换系统的字体大小产生的问题,如果我切换系统字体的样式呢?出现问题又该怎么解决?
其实有了上面的基础,解决这个问题就容易的多了。
在切换字体大小的时候,我们知道系统会发送通知,我们设置fontScale属性就可以了。但是切换字体的样式,系统并没有通知,谷歌官方也没有,stack上也是各种提问,没有找到监听的事件或者方法。
切换样式之后,我们可以观察一下Activity的生命周期,依然是onDestroy(),onCreate()….因为是异常的destroy,系统会把当前的状态信息存入saveInstanceState()方法里面的bundle,onCreate()的时候,从bundle里面取出来进行恢复。有时候,保存的bundle状态不对,导致界面加载错乱。
解决的方法: 在onCreate()的时候,判断bundle里面是否有数据,如果有数据,进行赋空处理,然后走一遍正常的流程,就可以了。具体代码如下:
@Override
protected void onCreate(Bundle savedInstanceState) {
**if (null != savedInstanceState)
savedInstanceState = null;**
super.onCreate(savedInstanceState);}
复制代码
这个配置可以放在项目的BaseActivity里面,可以减少对现有代码的改动。
思考:其实这个时候,你会发现,在切换字体大小的时候,也可以这样配置,甚至这样配置之后就不用管fontScale之类的属性了,让界面自己刷新去吧,反正第二次的onCreate()跟重新进入app效果是一样的。