浅谈android中的ListView之解决ScrollView和ListView嵌套冲突(实际上一切都是浮云,闲的蛋疼)(一)

本文主要是介绍浅谈android中的ListView之解决ScrollView和ListView嵌套冲突(实际上一切都是浮云,闲的蛋疼)(一),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

     相信大家都已经可以熟练使用ListView和GridView,大神们估计都在使用RecyclerView了。如果还在使用ListView,你肯定有这样的一个深刻的感受,那就是在做一个APP的时候使用ListView和GridView很频繁,并且经常会遇到一个页面中除了有ListView或GridView可能还有一些其他的内容,但是可能内容很多,你第一时间就会想到让它整体滑动即可,那就是在总的布局外面包裹一个ScrollView。也就是出现了ScrollView中嵌套一个ListView的场景,或者你的ScrollView嵌套多个ListView或者GridView的时候。我们自认为出现场景应该是整体内容会滑动,但是你会惊讶的发现并不是这样的,你会发现如果是嵌套了一个ListView或者GridView的时候,ListView只会显示一个Item项,GridView也会只显示一行。看来我们还是错了,具体是什么样子的下面我们通过一个demo来看看,然后分析一下为什么会出现这样的结果,以及最后解决的办法。

 我就简单写了一个ListView只要能说明问题即可,至于ListView的优化等问题不是本篇博客的重点,将会在后续的博客出现。

布局文件(注意:仔细看清楚ListView的layout高度的属性是warp_content)

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:id="@+id/parent_layout"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical" ><includeandroid:id="@+id/include_header_home"android:layout_width="match_parent"android:layout_height="wrap_content"layout="@layout/header_home" /><ScrollViewandroid:id="@+id/scrollview"android:layout_width="match_parent"android:layout_height="wrap_content" ><LinearLayoutandroid:layout_width="match_parent"android:layout_height="wrap_content"android:orientation="vertical" ><ImageViewandroid:layout_width="match_parent"android:layout_height="380dp"android:scaleType="fitXY"android:src="@drawable/center_image" /><LinearLayoutandroid:layout_width="match_parent"android:layout_height="wrap_content"android:orientation="vertical" ><LinearLayoutandroid:id="@+id/linear"android:layout_width="match_parent"android:layout_height="wrap_content"android:background="#FFF"android:orientation="horizontal" ><Viewandroid:layout_width="0.1dp"android:layout_height="match_parent"android:background="#44000000" /><TextViewandroid:layout_width="0dp"android:layout_height="wrap_content"android:layout_weight="1"android:drawableRight="@drawable/all_category_arrow_down"android:gravity="center"android:padding="15dp"android:text="商家分类" /><Viewandroid:layout_width="0.1dp"android:layout_height="match_parent"android:background="#44000000" /><TextViewandroid:id="@+id/test"android:layout_width="0dp"android:layout_height="wrap_content"android:layout_weight="1"android:drawableRight="@drawable/all_category_arrow_down"android:gravity="center"android:padding="15dp"android:text="智能排序" /><Viewandroid:layout_width="0.1dp"android:layout_height="match_parent"android:background="#77000000" /><TextViewandroid:id="@+id/wei_express"android:layout_width="0dp"android:layout_height="wrap_content"android:layout_weight="1"android:drawableRight="@drawable/slip_off_img_bd_express"android:gravity="center"android:padding="15dp"android:text="微送" /></LinearLayout><Viewandroid:layout_width="match_parent"android:layout_height="0.1dp"android:background="#44000000" /></LinearLayout><ListViewandroid:id="@+id/listview"android:layout_width="match_parent"android:layout_height="wrap_content"android:divider="#33000000"android:dividerHeight="0.1dp" ></ListView></LinearLayout></ScrollView></LinearLayout>

MainActivity.java

package com.mikyou.listviewtest;import com.mikyou.utils.SystemStatusManager;import android.app.Activity;
import android.database.DataSetObserver;
import android.os.Build;
import android.os.Bundle;
import android.view.View;
import android.view.ViewGroup;
import android.view.Window;
import android.view.WindowManager;
import android.widget.BaseAdapter;
import android.widget.ListAdapter;
import android.widget.ListView;public class MainActivity extends Activity {private ListView  mListView;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);requestWindowFeature(Window.FEATURE_NO_TITLE);setTranslucentStatus();setContentView(R.layout.activity_main);initData();initView();}private void initView() {mListView=(ListView) findViewById(R.id.listview);mListView.setAdapter(new MyAdapter());}class MyAdapter extends BaseAdapter{@Overridepublic int getCount() {return 7;}@Overridepublic Object getItem(int position) {return null;}@Overridepublic long getItemId(int position) {return position;}@Overridepublic View getView(int position, View convertView, ViewGroup parent) {View view=View.inflate(MainActivity.this, R.layout.home_list_item, null);return view;}}private void initData() {}private void setTranslucentStatus() {//沉浸标题栏效果// TODO Auto-generated method stubif(Build.VERSION.SDK_INT>=Build.VERSION_CODES.KITKAT){Window win=getWindow();WindowManager.LayoutParams winParams=win.getAttributes();final int bits=WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS;winParams.flags |=bits;win.setAttributes(winParams);}SystemStatusManager tintManager = new SystemStatusManager(this);tintManager.setStatusBarTintEnabled(true);tintManager.setStatusBarTintResource(0);tintManager.setNavigationBarTintEnabled(true);}
}


运行的结果:

注意:你会发现只会显示一个,而我的设置adapter的时候明明设置的是7个,为什么只显示一个呢??这是为什么,这就是意味ListView和ScrollView的嵌套带来的问题,因为ListView和GridView本身就是继承于ScrollView,而ScrollView中再次嵌套一个ScrollVewi就会出现高度测量的问题,我们都知道ListView也可以上下滑动,当ListView中的内容很多的时候,屏幕不足以显示的时候才会滑动显示。那为什么只会显示一项呢?其实仔细分析一下很简单就是ListView高度不够,如果高度达到一定的话,就会出现其他的Item项,注意看我们的布局ListView的Wrap_content,也就是测量里面的内容的高度来得到的,也就对应着android中的VIew测量的模式中的AT_MOST,这种测量模式实际上系统是不会给你真正的测的,而是根据测量其内部的内容来给出一个适合的尺寸,是系统只会测EXACTLY模式即为对应match_parent和指定明确的尺寸。我们还知道ListView每个Item加载模式通过getView方法一个一个的加载出来的,也就是当你第一个Item加载完后,它只测量第一个Item,所以只显示第一个Item,因为高度不够,只能显示第一个。如果不信,我们可以作如下的两个实验,第一打印出此时ListVIew的高度看看是不是此时ListView的高度是不是等于第一个Item的高度,第二就是我不指定warp_content,我指定一个明确尺寸看看。

第一个实验:比较第一个Item高度和ListVIew总高度,来说明此时ScrollView测量第一个Item来作为整个ListView的高度。

通过给ListView添加getViewTreeObserver().addOnGlobalLayoutListener事件,然后在回调方法中去得到高度,注意:因为我们直接或者间接在Activity中的OnCreate方法中去得到高度是为0,因为此时Activity中的View窗体树并没有绘制完毕,该方法就是通过监听整个View的树绘制完毕后才会回调,所以在该方法才能得到高度

package com.mikyou.listviewtest;import com.mikyou.utils.SystemStatusManager;import android.app.Activity;
import android.os.Build;
import android.os.Bundle;
import android.view.View;
import android.view.ViewGroup;
import android.view.ViewTreeObserver.OnGlobalLayoutListener;
import android.view.Window;
import android.view.WindowManager;
import android.widget.BaseAdapter;
import android.widget.ListView;public class MainActivity extends Activity {private ListView  mListView;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);requestWindowFeature(Window.FEATURE_NO_TITLE);setTranslucentStatus();setContentView(R.layout.activity_main);initData();initView();}private void initView() {mListView=(ListView) findViewById(R.id.listview);mListView.setAdapter(new MyAdapter());//mListView.getViewTreeObserver().addOnGlobalLayoutListener(new OnGlobalLayoutListener() {@Overridepublic void onGlobalLayout() {View firstItemView=mListView.getAdapter().getView(0, null, mListView);//得到第一个ItemfirstItemView.measure(0, 0);System.out.println("第一个Item的高度:"+firstItemView.getMeasuredHeight());System.out.println("mListView的高度:"+mListView.getHeight());				}});}class MyAdapter extends BaseAdapter{@Overridepublic int getCount() {return 7;}@Overridepublic Object getItem(int position) {return null;}@Overridepub

这篇关于浅谈android中的ListView之解决ScrollView和ListView嵌套冲突(实际上一切都是浮云,闲的蛋疼)(一)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1111973

相关文章

hdu1254(嵌套bfs,两次bfs)

/*第一次做这种题感觉很有压力,思路还是有点混乱,总是wa,改了好多次才ac的思路:把箱子的移动当做第一层bfs,队列节点要用到当前箱子坐标(x,y),走的次数step,当前人的weizhi(man_x,man_y),要判断人能否将箱子推到某点时要嵌套第二层bfs(人的移动);代码如下:

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

如何解决线上平台抽佣高 线下门店客流少的痛点!

目前,许多传统零售店铺正遭遇客源下降的难题。尽管广告推广能带来一定的客流,但其费用昂贵。鉴于此,众多零售商纷纷选择加入像美团、饿了么和抖音这样的大型在线平台,但这些平台的高佣金率导致了利润的大幅缩水。在这样的市场环境下,商家之间的合作网络逐渐成为一种有效的解决方案,通过资源和客户基础的共享,实现共同的利益增长。 以最近在上海兴起的一个跨行业合作平台为例,该平台融合了环保消费积分系统,在短

浅谈主机加固,六种有效的主机加固方法

在数字化时代,数据的价值不言而喻,但随之而来的安全威胁也日益严峻。从勒索病毒到内部泄露,企业的数据安全面临着前所未有的挑战。为了应对这些挑战,一种全新的主机加固解决方案应运而生。 MCK主机加固解决方案,采用先进的安全容器中间件技术,构建起一套内核级的纵深立体防护体系。这一体系突破了传统安全防护的局限,即使在管理员权限被恶意利用的情况下,也能确保服务器的安全稳定运行。 普适主机加固措施:

Android平台播放RTSP流的几种方案探究(VLC VS ExoPlayer VS SmartPlayer)

技术背景 好多开发者需要遴选Android平台RTSP直播播放器的时候,不知道如何选的好,本文针对常用的方案,做个大概的说明: 1. 使用VLC for Android VLC Media Player(VLC多媒体播放器),最初命名为VideoLAN客户端,是VideoLAN品牌产品,是VideoLAN计划的多媒体播放器。它支持众多音频与视频解码器及文件格式,并支持DVD影音光盘,VCD影

android-opencv-jni

//------------------start opencv--------------------@Override public void onResume(){ super.onResume(); //通过OpenCV引擎服务加载并初始化OpenCV类库,所谓OpenCV引擎服务即是 //OpenCV_2.4.3.2_Manager_2.4_*.apk程序包,存

pip-tools:打造可重复、可控的 Python 开发环境,解决依赖关系,让代码更稳定

在 Python 开发中,管理依赖关系是一项繁琐且容易出错的任务。手动更新依赖版本、处理冲突、确保一致性等等,都可能让开发者感到头疼。而 pip-tools 为开发者提供了一套稳定可靠的解决方案。 什么是 pip-tools? pip-tools 是一组命令行工具,旨在简化 Python 依赖关系的管理,确保项目环境的稳定性和可重复性。它主要包含两个核心工具:pip-compile 和 pip

【VUE】跨域问题的概念,以及解决方法。

目录 1.跨域概念 2.解决方法 2.1 配置网络请求代理 2.2 使用@CrossOrigin 注解 2.3 通过配置文件实现跨域 2.4 添加 CorsWebFilter 来解决跨域问题 1.跨域概念 跨域问题是由于浏览器实施了同源策略,该策略要求请求的域名、协议和端口必须与提供资源的服务相同。如果不相同,则需要服务器显式地允许这种跨域请求。一般在springbo

从状态管理到性能优化:全面解析 Android Compose

文章目录 引言一、Android Compose基本概念1.1 什么是Android Compose?1.2 Compose的优势1.3 如何在项目中使用Compose 二、Compose中的状态管理2.1 状态管理的重要性2.2 Compose中的状态和数据流2.3 使用State和MutableState处理状态2.4 通过ViewModel进行状态管理 三、Compose中的列表和滚动

Android 10.0 mtk平板camera2横屏预览旋转90度横屏拍照图片旋转90度功能实现

1.前言 在10.0的系统rom定制化开发中,在进行一些平板等默认横屏的设备开发的过程中,需要在进入camera2的 时候,默认预览图像也是需要横屏显示的,在上一篇已经实现了横屏预览功能,然后发现横屏预览后,拍照保存的图片 依然是竖屏的,所以说同样需要将图片也保存为横屏图标了,所以就需要看下mtk的camera2的相关横屏保存图片功能, 如何实现实现横屏保存图片功能 如图所示: 2.mtk