横竖屏切换导致页面频繁重启screenLayout解析

 更新时间:2023年03月15日 11:25:50   作者:BennuCTech  
这篇文章主要为大家介绍了横竖屏切换导致页面频繁重启screenLayout解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

前言

前几天多名用户反馈同一个问题,在小新平板上无法上网课,点击上课按钮后就退回到首页了。同事了解了一下发现小新平板现在销量特别好,于是赶紧申请了一台测试机打算看看到底是什么问题。

最后同事发现是screenLayout的问题,在manifest中为需要横竖屏切换的Acitivty配置screenLayout即可,如下:

<activity android:name=".MainActivity"
  android:configChanges="orientation|keyboardHidden|screenSize|screenLayout"
    android:launchMode="singleTask">
    <intent-filter>
       ...
    </intent-filter>
</activity>

我们之前android:configChanges配置是orientation|keyboardHidden|screenSize,缺少了screenLayout。

但是为什么在其他设备上没问题,唯独在小新平板上有问题呢?而且为什么添加了screenLayout就解决问题了,这其中的原理是什么?我非常好奇,于是自己研究了一下。

android:configChanges

首先我们要知道android:configChanges这个配置的作用,这里我们来看看官方的介绍:

列出 Activity 将自行处理的配置变更。在运行时发生配置变更时,默认情况下会关闭 Activity 并将其重启,但使用该属性声明配置将阻止 Activity 重启。相反,Activity 会保持运行状态,并且系统会调用其 onConfigurationChanged() 方法。

当Activity的配置发生变更时(如横竖屏切换),如果在android:configChanges中没有添加该配置,那么就会关闭并重启Activity,这时候debug会发现重新执行了onCreate。但是当我们添加了该配置,如果该配置发生变更,则不会重启Activity,会调用onConfigurationChanged()方法。

那么orientation就是对应着横竖屏切换,keyboardHidden则是软键盘弹出,screenSize则是屏幕尺寸改变。这么来看我们设置了orientation应该就可以了,但是官方在这里有一个提示,如下:

官方建议在设置orientation的同时设置screenSizescreenLayoutscreenSize我么可以理解,横竖屏切换时宽高会交换,那么screenLayout是指什么呢?

screenLayout

前面我们知道存在配置时会执行onConfigurationChanged(),这个函数的参数是Configuration类型的,这个类里保存着Activity的配置,我们来看看对screenLayout这个属性的描述:

/**
Bit mask of overall layout of the screen. Currently there are four fields:
The SCREENLAYOUT_SIZE_MASK bits define the overall size of the screen. 
They may be one of SCREENLAYOUT_SIZE_SMALL, SCREENLAYOUT_SIZE_NORMAL, 
SCREENLAYOUT_SIZE_LARGE, or SCREENLAYOUT_SIZE_XLARGE.
The SCREENLAYOUT_LONG_MASK defines whether the screen is wider/taller than normal. 
They may be one of SCREENLAYOUT_LONG_NO or SCREENLAYOUT_LONG_YES.
The SCREENLAYOUT_LAYOUTDIR_MASK defines whether the screen layout is either LTR or RTL. 
They may be one of SCREENLAYOUT_LAYOUTDIR_LTR or SCREENLAYOUT_LAYOUTDIR_RTL.
The SCREENLAYOUT_ROUND_MASK defines whether the screen has a rounded shape. 
They may be one of SCREENLAYOUT_ROUND_NO or SCREENLAYOUT_ROUND_YES.
See Supporting Multiple Screens for more information.
**/
public int screenLayout;

可以看到screenLayout其实是承载着四个配置的:

  • 屏幕大小等级:有SCREENLAYOUT_SIZE_SMALL, SCREENLAYOUT_SIZE_NORMAL, SCREENLAYOUT_SIZE_LARGESCREENLAYOUT_SIZE_XLARGE四种
  • 是否宽屏:屏幕是否比普通屏幕更宽或更高
  • 屏幕方向:屏幕是从左向右显示,还是从有向左显示
  • 是否是圆角屏:屏幕是否有圆角

通过将screenLayout于对应mask进行与运算就可以得到当前屏幕在该属性的值,比如:

screenLayout & SCREENLAYOUT_SIZE_MASK

就可以得到屏幕大小等级,一定是SCREENLAYOUT_SIZE_SMALL, SCREENLAYOUT_SIZE_NORMAL, SCREENLAYOUT_SIZE_LARGESCREENLAYOUT_SIZE_XLARGE之一。

四个mask对应的二进制分别是:

  • SCREENLAYOUT_SIZE_MASK: 0000 0000 1111
  • SCREENLAYOUT_LONG_MASK: 0000 0011 0000
  • SCREENLAYOUT_LAYOUTDIR_MASK:0000 1100 0000
  • SCREENLAYOUT_ROUND_MASK: 0011 0000 0000

比如在我的测试机上得到的screenLayout是268435810,转成二进制就是

0001 0000 0000 0000 0000 0001 0110 0010

通过与四个mask分别计算得到SCREENLAYOUT_SIZE_NORMAL、SCREENLAYOUT_LONG_NO、SCREENLAYOUT_LAYOUTDIR_LTR和SCREENLAYOUT_ROUND_NO

而且无论横屏还是竖屏,我的测试机的screenLayout是不变的,所以即使在android:configChanges中没用添加screenLayout,横竖屏切换的时候也不会重启Activity,因为其他两个属性orientationscreenSize都添加了。

小新平板

那么为什么小新平板上会有不同的现象,在android:configChanges中添加screenLayout完后我在onConfigurationChanged函数中debug获取screenLayout值,发现横屏和竖屏这个值是不同的,分别是268435812和268435796。

通过与mask运算获取四个属性后对比发现,横竖屏切换后SCREENLAYOUT_LONG_NO变成了SCREENLAYOUT_LONG_YES,所以没添加screenLayout的时候就会导致Activity重启,添加后就可以了。

至于为什么重启Activity会导致回退到首页,其实是用户表述问题,现象是应用重启了。为什么应用会重启,这是因为我们的应用架构是单Activity的,页面由fragment承载。当从详情页打开上课页面时,会通过代码手动将竖屏切换到横屏(同时为了返回详情页时换回竖屏,在详情页手动切换回竖屏)。这时候Activity重启并以横屏状态恢复所有fragment,但是恢复详情页时候,又切换到竖屏,所以又重启;然后以竖屏状态恢复到上课页面,又进行了切换,于是死循环,最后系统将应用重启。

总结

一直以来对android:configChanges没有深入了解,正好趁着这个机会了解了一番,大家以后一定要注意这里,特别注意官方文档的提示,很重要。

以上就是横竖屏切换导致页面频繁重启screenLayout解析的详细内容,更多关于screenLayout页面重启的资料请关注脚本之家其它相关文章!

相关文章

  • Android自定义View仿QQ运动步数效果

    Android自定义View仿QQ运动步数效果

    这篇文章主要为大家详细介绍了Android自定义View仿QQ运动步数效果,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-11-11
  • 深入Android SQLite 事务处理详解

    深入Android SQLite 事务处理详解

    本篇文章是对Android中SQLite 事务处理进行了详细的分析介绍。需要的朋友参考下
    2013-05-05
  • Android自定义视图实现手指移动轨迹

    Android自定义视图实现手指移动轨迹

    这篇文章主要为大家详细介绍了Android自定义视图实现手指移动轨迹,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-06-06
  • Android仿百度地图小度语音助手的贝塞尔曲线动画

    Android仿百度地图小度语音助手的贝塞尔曲线动画

    这篇文章主要为大家详细介绍了Android仿百度地图小度语音助手的贝塞尔曲线动画,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-07-07
  • Android自定义View实现loading动画加载效果

    Android自定义View实现loading动画加载效果

    项目开发中对Loading的处理是比较常见的,安卓系统提供的不太美观,引入第三发又太麻烦,这时候自己定义View来实现这个效果。这篇文章主要介绍了Android自定义View实现loading动画加载效果,需要的朋友可以参考下
    2017-03-03
  • Android View的事件体系教程详解

    Android View的事件体系教程详解

    这篇文章主要为大家介绍了Android View的事件体系教程详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步早日升职加薪
    2022-03-03
  • Android中利用viewflipper动画切换屏幕效果

    Android中利用viewflipper动画切换屏幕效果

    这篇文章主要介绍了Android中利用viewflipper动画切换屏幕效果的相关资料,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2016-09-09
  • 深入浅出学习Android ListView基础

    深入浅出学习Android ListView基础

    这篇文章主要介绍了深入浅出的带领大家学习Android ListView基础,ListView是安卓里常用的控件,本文介绍一下常用用法,以及优化等方法,感兴趣的小伙伴们可以参考一下
    2016-01-01
  • Android RecyclerView基本使用详解

    Android RecyclerView基本使用详解

    这篇文章主要为大家详细介绍了Android RecyclerView基本使用的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-02-02
  • Android实现可滑动的自定义日历控件

    Android实现可滑动的自定义日历控件

    这篇文章主要为大家详细介绍了Android实现可滑动的自定义日历控件,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-07-07

最新评论