从原理上说说ScrollView嵌套ListView的问题-阿里云开发者社区

开发者社区> sydmobile> 正文

从原理上说说ScrollView嵌套ListView的问题

简介: 版权声明:本文为sydMobile原创文章,转载请务必注明出处! https://blog.csdn.net/sydMobile/article/details/79543986 文章最早发布于我的微信公众号 Android_De_Home 中,欢迎大家扫描下面二维码关注微信公众获取更多知识内容。
+关注继续查看
版权声明:本文为sydMobile原创文章,转载请务必注明出处! https://blog.csdn.net/sydMobile/article/details/79543986

文章最早发布于我的微信公众号 Android_De_Home 中,欢迎大家扫描下面二维码关注微信公众获取更多知识内容。
本文为sydMobile原创文章,可以随意转载,但请务必注明出处!

ScrollView嵌套ListView会出现的问题,相信大家已经见到的非常多了,对于解决方法也是了如指掌了。但是原理你清楚了吗?这里主要讲为什么会出现这种问题,已经解决这个问题的原理。

ScrollView嵌套ListView出现的问题

ScrollView嵌套ListView会出现的问题,相信大家都已经见的非常多了,对于怎么解决也不陌生了。

这里再来说一下:
出现的问题:
ListView会显示不全

解决方法:
最常见的一种方法:

自己继承ListView,重写onMeasure()方法

    @Override
    public void onMeasure(int widthMeasureSpec,int heightMeasureSpec){
        super.onMeasure(widthMeasureSpec,MeasureSpec.makeMeasureSpec
        (Integer.MAX_VALUe)>>2,MeasureSpec.AT_MOST));

    }

一般的我们仅仅需要这样重写这个方法就可以很顺利的解决ScrollView嵌套ListView出现的ListView显示不全的问题了。那么是因为什么呢?下面我们就从原理上说说!

解决原理

说起原理就可MeasureSpec类分不开了,先来介绍一下这个类。

MeasureSpec类

MeasureSpec类是View的一个内部静态类,MeasureSpec类封装了从父布局到子布局传递的布局需求。每个MeasureSpec对象代表了宽度和高度的要求。一个MeasureSpec类的表示由控件大小和模式两组成。有三种模式:

  • UNSPECIFIED
    父布局没有对子布局施加任何的限制,子布局可以是任何他想要的大小
  • EXACTLY
    父布局已经确定了子布局的大小,子布局会在父布局给出的界限内。子布局的大小是精确的。父布局给多大就是多大。
  • AT_MOST
    父布局会给定一个最大的值,子布局的大小是不能超过这个值的。但是可以比这个值小。

MeasureSpec类为了减少对象的分配用了一个整数来实现这个功能(父布局传递到子布局的的布局需求),这个整数是用模式和大小来表示。

那么这个整数是怎么来实现这个功能的呢?

int表示形式

我们都知道int类型的是32位,那么表示形式就是,向上面图中的那样,前两位代表了模式(就是前面提到的那三种),后30位代表了组件的大小。这样就用整数形式来表示模式和大小了。

具体的看一下三种模式

UNSPECIFIED 模式
UNSPECIFIED == 0 << MODE_SHIFT 也就是 0 向左位移30位,结果就是int类型的最高位是 00

EXACTLY 模式
EXACTLY == 1 << MODE_SHIFT ;也就是 01向左位移30位,结果就是int类型的最高两位是 01

AT_MOST 模式
AT_MOST = 2 << MODE_SHIFT ;也就是 10 向左位移30位,结果就是int类型的最高两位是 10

makeMeasureSpec()方法

在这个MeasureSpec类中最重要的一个方法恐怕就是makeMeasureSpec这个方法了。
makeMeasureSpec

这个方法就是用给定的大小和模式创建一个int类型的数来满足父布局到子布局传递的布局需求。第一个参数 size就是父布局给子布局传递的大小,第二个参数是模式(就是在上面的三个模式中选择一个)。好了,到这里makeMeasureSpec()这个方法也讲了。

ScrollView嵌套ListView解决方法

方法一:

上面已经讲了,重写ListView的onMeasure()方法


    @Override
    protected void onMeasure(int widthMeasureSpec,int heightMeasureSpec){
            super.onMeasure(widthMeasureSpec,MeasureSpec.makeMeasureSpec(Integer

            .MAX_VALUE >> 2,MeasureSpec.AT_MOST));
    }

我们在修改的时候并没有改变widthMeasureSpec,仅仅是修改了heightMeasureSpec,因为ScrollView设置成了上下滑动,横向并没有滑动,所有在横向上并没有和ListView产生冲突。所以传入的widthMeasureSpec是正确的,而heightMeasureSpec是不正确的,因为ListView嵌套在ScrollView中,也就是说ScrollView是ListView的子布局,这个时候他们的滑动事件发生了冲突,这个值也就不正确了,不是LIstView的实际高度。所以我们要重写传入height,第一个参数为什么是Integer.MAX_VALUE >>2 呢 ?我们说了MeasureSpec用 int类型表示前两位代表模式,后30位代表大小,我们就需要让后面30位是int类型中最大的值就可以了。为什么选择AT_MOST模式呢?这个模式是父布局给定一个值,不能超过这个值,我们很显然已经给了最大值了。

方法二:

既然测不出高度,那么我就手中在代码中设置ListView的高度。

    public static void setListViewHeightBasedOnChildren(ListView listView) {
        if(listView == null) return;

        ListAdapter listAdapter = listView.getAdapter();
        if (listAdapter == null) {
            // pre-condition
            return;
        }

        int totalHeight = 0;
        for (int i = 0; i < listAdapter.getCount(); i++) {
            View listItem = listAdapter.getView(i, null, listView);
            listItem.measure(0, 0);
            totalHeight += listItem.getMeasuredHeight();
        }

        ViewGroup.LayoutParams params = listView.getLayoutParams();
        params.height = totalHeight + (listView.getDividerHeight() * (listAdapter.getCount() - 1));
        listView.setLayoutParams(params);
}

这种方法有前提条件限制:
Adapter中getView方法返回的View的必须由LinearLayout组成,因为只有LinearLayout才有measure()方法,如果使用其他的布局如RelativeLayout,在调用listItem.measure(0, 0);时就会抛异常,因为除LinearLayout外的其他布局的这个方法就是直接抛异常!

总结 自定义ListView比较好用,还有一个问题就是无论使用上面哪一个方法,当你的ListView在加载数据的时候,如果当前页面没展示完全,那么scrollView会自动往下滑动一点,也就是造成了你进入这个页面的时候,默认页面是往下滑动了一下,而不是在最顶端。 造成这个问题主要的原因还是焦点问题,ListView默认情况下,isFocusableInTouchMode和isFocusable都是false的,但是当在加载数据后这两个值就会变为true了。如果在布局中没有其他view获取焦点,这个时候ListView就争夺到了焦点,也就造成了滑动。

这个问题的解决方法:
1.在加载完数据后设置ScrollView滑动到顶部scrollView.smoothScrollTo(0,0)
这种做法是有缺点的,你会看到屏幕滑动一下。

2.使用descendantFousability属性

descendantFocusability有三种属性
beforeDescendant:viewgroup会优先其子类控件而获取到焦点。
afterDescendant:viewgroup只有当其子类控件不需要获取焦点的时候才获取焦点。
blocksDescendants: viewgroup会覆盖子类控件而直接获得焦点

在ScrollView的LinearLayout中添加android:denscendantFocusability = “blocksDescendants”就可以了。

欢迎大家关注我的微信公众号,和我交流分享

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
使用NAT网关轻松为单台云服务器设置多个公网IP
在应用中,有时会遇到用户询问如何使单台云服务器具备多个公网IP的问题。 具体如何操作呢,有了NAT网关这个也不是难题。
26763 0
阿里云服务器怎么设置密码?怎么停机?怎么重启服务器?
如果在创建实例时没有设置密码,或者密码丢失,您可以在控制台上重新设置实例的登录密码。本文仅描述如何在 ECS 管理控制台上修改实例登录密码。
9980 0
阿里云服务器如何登录?阿里云服务器的三种登录方法
购买阿里云ECS云服务器后如何登录?场景不同,阿里云优惠总结大概有三种登录方式: 登录到ECS云服务器控制台 在ECS云服务器控制台用户可以更改密码、更换系.
13752 0
windows server 2008阿里云ECS服务器安全设置
最近我们Sinesafe安全公司在为客户使用阿里云ecs服务器做安全的过程中,发现服务器基础安全性都没有做。为了为站长们提供更加有效的安全基础解决方案,我们Sinesafe将对阿里云服务器win2008 系统进行基础安全部署实战过程! 比较重要的几部分 1.
9139 0
腾讯云服务器 设置ngxin + fastdfs +tomcat 开机自启动
在tomcat中新建一个可以启动的 .sh 脚本文件 /usr/local/tomcat7/bin/ export JAVA_HOME=/usr/local/java/jdk7 export PATH=$JAVA_HOME/bin/:$PATH export CLASSPATH=.
4646 0
阿里云ECS云服务器初始化设置教程方法
阿里云ECS云服务器初始化是指将云服务器系统恢复到最初状态的过程,阿里云的服务器初始化是通过更换系统盘来实现的,是免费的,阿里云百科网分享服务器初始化教程: 服务器初始化教程方法 本文的服务器初始化是指将ECS云服务器系统恢复到最初状态,服务器中的数据也会被清空,所以初始化之前一定要先备份好。
7317 0
阿里云服务器ECS登录用户名是什么?系统不同默认账号也不同
阿里云服务器Windows系统默认用户名administrator,Linux镜像服务器用户名root
4449 0
+关注
sydmobile
热爱分享,喜欢钻研技术。希望为开源世界贡献自己的一份力量。
38
文章
0
问答
文章排行榜
最热
最新
相关电子书
更多
《2021云上架构与运维峰会演讲合集》
立即下载
《零基础CSS入门教程》
立即下载
《零基础HTML入门教程》
立即下载