【转】Android开发的那些坑和小技巧

本文转自: LeoLiang的Blog

ps: 排版还有问题, 将就下, 实在看不惯就去原文瞅瞅吧~   其实有非常多的坑和小技巧可供分享, 在此转载, 同时未来可能会结合自身开发中的坑, 加上之前有心人总结的帖子,一并在此更新一下, 链接未来添加, 今天事情还不少, 就暂且占个萝卜坑先 ~

1、android:clipToPadding


意思是控件的绘制区域是否在padding里面。默认为true。如果你设置了此属性值为false,就能实现一个在布局上事半功陪的效果。先看一个效果图。


上图中的ListView顶部默认有一个间距,向上滑动后,间距消失,如下图所示。

如果使用margin或padding,都不能实现这个效果。加一个headerView又显得大材小用,而且过于麻烦。此处的clipToPadding配合paddingTop效果就刚刚好。

2、match_parent和wrap_content


按理说这两个属性一目了然,一个是填充布局空间适应父控件,一个是适应自身内容大小。但如果在列表如ListView中,用错了问题就大了。ListView中的getView方法需要计算列表条目,那就必然需要确定ListView的高度,onMesure才能做测量。如果指定了wrap_content,就等于告诉系统,如果我有一万个条目,你都帮我计算显示出来,然后系统按照你的要求就new了一万个对象出来。那你不悲剧了?先看一个图。

假设现在ListView有8条数据,match_parent需要new出7个对象,而wrap_content则需要8个。这里涉及到View的重用,就不多探讨了。所以这两个属性的设置将决定getView的调用次数。

由此再延伸出另外一个问题:getView被多次调用

什么叫多次调用?比如position=0它可能调用了几次。看似很诡异吧。GridView和ListView都有可能出现,说不定这个祸首就是wrap_content。说到底是View的布局出现了问题。如果嵌套的View过于复杂,解决方案可以是通过代码测量列表所需要的高度,或者在getView中使用一个小技巧:parent.getChildCount == position

        @Override

        public  View getView(intposition, View convertView, ViewGroup parent) {

               if(parent.getChildCount() ==position) {

                //does things here

               }

                returnconvertView;

        }

3、IllegalArgumentException: pointerIndex out of range


出现这个Bug的场景还是很无语的。一开始我用ViewPager + PhotoView(一个开源控件)显示图片,在多点触控放大缩小时就出现了这个问题。一开始我怀疑是PhotoView的bug,找了半天无果。要命的是不知如何try,老是crash。后来才知道是android遗留下来的bug,源码里没对pointer index做检查。改源码重新编译不太可能吧。明知有exception,又不能从根本上解决,如果不让它crash,那就只能try-catch了。解决办法是:自定义一个ViewPager并继承ViewPager。请看以下代码:

/*** 自定义封装android.support.v4.view.ViewPager,重写onInterceptTouchEvent事件,捕获系统级别异常*/

         public  class  CustomViewPager  extends  ViewPager {

                  public  CustomViewPager(Context context) {this(context,null);

                  }

                  public  CustomViewPager(Context context, AttributeSet attrs) {super(context, attrs);

                  }

                 @Override

                 public  boolean  onInterceptTouchEvent(MotionEvent ev) {try{returnsuper.onInterceptTouchEvent(ev);

                  }catch(IllegalArgumentException e) {

                      LogUtil.e(e);

                  }catch(ArrayIndexOutOfBoundsException e) {

                     LogUtil.e(e);

                  }return  false;

                }

         }

把用到ViewPager的布局文件,替换成CustomViewPager就OK了。

4、ListView中item点击事件无响应


listView的Item点击事件突然无响应,问题一般是在listView中加入了button、checkbox等控件后出现的。这个问题是聚焦冲突造成的。在android里面,点击屏幕之后,点击事件会根据你的布局来进行分配的,当你的listView里面增加了button之后,点击事件第一优先分配给你listView里面的button。所以你的点击Item就失效了,这个时候你就要根据你的需求,是给你的item的最外层layout设置点击事件,还是给你的某个布局元素添加点击事件了。

解决办法:在ListView的根控件中设置(若根控件是LinearLayout, 则在LinearLayout中加入以下属性设置)descendantFocusability属性。

android:descendantFocusability="blocksDescendants"

官方文档也是这样说明。

5、getSupportFragmentManager()和getChildFragmentManager()


有一个需求,Fragment需要嵌套3个Fragment。基本上可以想到用ViewPager实现。开始代码是这样写的:

mViewPager.setAdapter(new  CustomizeFragmentPagerAdapter(getActivity().getSupportFragmentManager(), subFragmentList));

导致的问题是嵌套的Fragment有时会莫名其妙不显示。开始根本不知道问题出现在哪,当你不知道问题的原因时,去解决这个问题显然比较麻烦。经过一次又一次的寻寻觅觅,终于在stackoverflow上看到了同样的提问。说是用getChildFragmentManager()就可以了。真是这么神奇!

mViewPager.setAdapter(new  CustomizeFragmentPagerAdapter(getChildFragmentManager, subFragmentList));

让我们看一下这两个有什么区别。首先是getSupportFragmentManager(或者getFragmentManager)的说明:

Return the FragmentManager for interacting with fragments associated with this fragment's activity.

然后是getChildFragmentManager:

Return aprivate FragmentManager for placing and managing Fragments inside of this Fragment.

Basically, the difference is that Fragment's now have their own internal FragmentManager that can handle Fragments. The child FragmentManager is the one that handles Fragments contained within only the Fragment that it was added to. The other FragmentManager is contained within the entire Activity.

已经说得比较明白了。

6、ScrollView嵌套ListView


这样的设计是不是很奇怪?两个同样会滚动的View居然放到了一起,而且还是嵌套的关系。曾经有一个这样的需求:界面一共有4个区域部分,分别是公司基本信息(logo、名称、法人、地址)、公司简介、公司荣誉、公司口碑列表。每部分内容都需要根据内容自适应高度,不能写死。鄙人首先想到的也是外部用一个ScrollView包围起来。然后把这4部分分别用4个自定义控件封装起来。基本信息和公司简介比较简单,荣誉需要用到RecyclerView和TextView的组合,RecyclerView(当然,用GridView也可以,3列多行的显示)存放荣誉图片,TextView显示荣誉名称。最后一部分口碑列表当然是ListView了。这时候,问题就出来了。需要解决ListView放到ScrollView中的滑动问题和RecyclerView的显示问题(如果RecyclerView的高度没法计算,你是看不到内容的)。

当然,网上已经有类似的提问和解决方案了。

给一个网址:

四种方案解决ScrollView嵌套ListView问题

ListView的情况还比较好解决,优雅的做法无非写一个类继承ListView,然后重写onMeasure方法。

         @Override

         protected  void  onMeasure(intwidthMeasureSpec,intheightMeasureSpec) {intexpandSpec =                                                                     MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2,                                                                                                       MeasureSpec.AT_MOST);super.onMeasure(widthMeasureSpec,   expandSpec);

         }

ListView可以重写onMeasure解决,RecyclerView重写这个方法是行不通的。

说到底其实计算高度嘛。有两种方式,一种是动态计算RecycleView,然后设置setLayoutParams;另外一种跟ListView的解决方式类似,定义一个类继承LinearLayoutManager或GridLayoutManager(注意:可不是继承RecyclerView),重写onMeasure方法(此方法比较麻烦,此处不表,下次写一篇文章再作介绍)。

动态计算高度如下:

             int  heightPx = DensityUtil.dip2px(getActivity(), (imageHeight + imageRowHeight) *lines);

             MarginLayoutParams mParams=newMarginLayoutParams(LayoutParams.MATCH_PARENT, heightPx);

             mParams.setMargins(0, 0, 0, 0);

             LinearLayout.LayoutParams lParams=newLinearLayout.LayoutParams(mParams);

             honorImageRecyclerView.setLayoutParams(lParams);

思路是这样的:服务端返回荣誉图片后,由于是3列显示的方式,只需要计算需要显示几行,然后给定行间距和图片的高度,再设置setLayoutParams就行了。

             int  lines = (int) Math.ceil(totalImages / 3d);

至此,这个奇怪的需求得到了解决。

可是在滑动的时候,感觉出现卡顿的现象。聪明的你肯定想到是滑动冲突了。应该是ScrollView的滑动干扰到了ListView的滑动。怎么办呢?能不能禁掉ScrollView的滑动?

百度一下,你肯定能搜索到答案的。先上代码:

/***@authorLeo

*

*        Created in 2015-9-12

*        拦截ScrollView滑动事件*/publicclassCustomScrollViewextendsScrollView {privateintdownY;privateinttouchSlop;publicCustomScrollView(Context context) {this(context,null);

}publicCustomScrollView(Context context, AttributeSet attrs) {this(context, attrs, 0);

}publicCustomScrollView(Context context, AttributeSet attrs,intdefStyleAttr) {super(context, attrs, defStyleAttr);

touchSlop=ViewConfiguration.get(context).getScaledTouchSlop();

}

@OverridepublicbooleanonInterceptTouchEvent(MotionEvent e) {intaction =e.getAction();switch(action) {caseMotionEvent.ACTION_DOWN:

downY= (int) e.getRawY();break;caseMotionEvent.ACTION_MOVE:intmoveY = (int) e.getRawY();if(Math.abs(moveY - downY) >touchSlop) {returntrue;

}

}returnsuper.onInterceptTouchEvent(e);

}

}

只要理解了getScaledTouchSlop()这个方法就好办了。这个方法的注释是:Distance in pixels a touch can wander before we think the user is scrolling。说这是一个距离,表示滑动的时候,手的移动要大于这个距离才开始移动控件,如果小于此距离就不触发移动。

看似很完美了。

但是还有另外一个问题:我每次加载这个界面花的时间太长了,每次由其它界面启动这个界面时,都要卡上1~2秒,而且因手机性能时间不等。并不是由于网络请求,取数据由子线程做,跟UI线程毫无关系。这样的体验自己看了都很不爽。

几天过去了,还是那样。马上要给老板演示了。这样的体验要被骂十次呀。

难道跟ScrollView的嵌套有关?

好吧,那我重构代码。不用ScrollView了。直接用一个ListView,然后add一个headerView存放其它内容。因为控件封装得还算好,没改多少布局就OK了,一运行,流畅顺滑,一切迎刃而解!

本来就是这么简单的问题,为什么非得用ScrollView嵌套呢?

stackoverflow早就告诉你了,不要这样嵌套!不要这样嵌套!不要这样嵌套!重要的事情说三遍。

ListView inside ScrollView is not scrolling on Android

当然,从android 5.0 Lollipop开始提供了一种新的API支持嵌入滑动,此时,让像这样的需求也能很好实现。

此处给一个网址,大家有兴趣自行了解,此处不再讨论。

Android NestedScrolling 实战

7、EmojiconTextView的setText(null)


这是开源表情库com.rockerhieu.emojicon中的TextView加强版。相信很多人用到过这个开源工具包。TextView用setText(null)完全没问题。但EmojiconTextView setText(null)后就悲剧了,直接crash,显示的是null pointer。开始我怀疑时这个view没初始化,但并不是。那就调试一下呗。

@OverridepublicvoidsetText(CharSequence text, BufferType type) {

SpannableStringBuilder builder=newSpannableStringBuilder(text);

EmojiconHandler.addEmojis(getContext(), builder, mEmojiconSize);super.setText(builder, type);

}

EmojiconTextView中的setText看来没什么问题。点SpannableStringBuilder进去看看,源码原来是这样的:

/*** Create a new SpannableStringBuilder containing a copy of the

* specified text, including its spans if any.*/publicSpannableStringBuilder(CharSequence text) {this(text, 0, text.length());

}

好吧。问题已经找到了,text.length(),不空指针才怪。

text = text ==null? "": text;

SpannableStringBuilder builder=newSpannableStringBuilder(text);

加一行判断就行了。

8、cursor.close()


一般来说,database的开和关不太会忘记,但游标的使用可能并不会引起太多重视,尤其是游标的随意使用。比如用ContentResolver结合Cursor查询SD卡中图片,很容易写出以下的代码:

Cursor cursor = contentResolver.query(uri,null, MediaStore.Images.Media.MIME_TYPE + "=? or "                        + MediaStore.Images.Media.MIME_TYPE + "=?",newString[] { "image/jpeg", "image/png"},

MediaStore.Images.Media.DATE_MODIFIED);while(cursor.moveToNext()) {//TODO}

cursor都不做非空判断,而且往往在关闭游标的时候不注意有可能异常抛出。

以前在项目中,经常出现由于游标没及时关闭或关闭出异常没处理好导致其它的问题产生,而且问题看起来非常的诡异,不好解决。后来,我把整个项目中有关游标的使用重构一遍,后来就再没发生过类似的问题。

原则很简单,所有Cursor的声明为:

Cursor cursor =null;

且放在try-catch外面;需要用到cursor,先做非空判断。然后在方法的最后用一个工具类处理游标的关闭。


/***@authorLeo

*

*        Created in 2015-9-15*/publicclassIOUtil {privateIOUtil() {

}publicstaticvoidcloseQuietly(Closeable closeable) {if(closeable !=null) {try{

closeable.close();

}catch(Throwable e) {

}

}

}publicstaticvoidcloseQuietly(Cursor cursor) {if(cursor !=null) {try{

cursor.close();

}catch(Throwable e) {

}

}

}

}

我想,这样就没必要在每个地方都try-catch-finally了。

先想到这么多,以后再补充。

参考:

android:clipToPadding和android:clipChildren

HowTo: ListView, Adapter, getView and different list items’ layouts in one ListView

android ListView 在初始化时多次调用getView()原因分析

java.lang.IllegalArgumentException: pointerIndex out of range Exception - dispatchTouchEvent

What is difference between getSupportFragmentManager() and getChildFragmentManager()?

标签:Android,ScrollView嵌套ListView

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,378评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,356评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,702评论 0 342
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,259评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,263评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,036评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,349评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,979评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,469评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,938评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,059评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,703评论 4 323
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,257评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,262评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,485评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,501评论 2 354
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,792评论 2 345

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,510评论 25 707
  • afinalAfinal是一个android的ioc,orm框架 https://github.com/yangf...
    passiontim阅读 15,401评论 2 45
  • ¥开启¥ 【iAPP实现进入界面执行逐一显】 〖2017-08-25 15:22:14〗 《//首先开一个线程,因...
    小菜c阅读 6,358评论 0 17
  • 愿你我都能清醒的活… 晨起,我硬生生地摆脱起床气跑了几公里。期间听张德芬小时空修心课,一本迈克·辛格的《清醒的活》...
    夏沫er阅读 223评论 0 1
  • 本月的主题是"没有放弃"!必须做的事情坚持起来不难,可做可不做的事情坚持起来才难!本月(其实是后半...
    静静是我啦阅读 205评论 0 0