Android开发中StackOverflowError错误实例分析

一、概述

我在一个复杂的layout嵌套较多的android界面,碰到了java.lang.StackOverflowError这个Fatal Exception,app程序crash退出。这个错误出现的比较奇怪,在我做技术调研的时候,这个界面是放在单独的一个程序中展示的,工作很正常,没有出现这个严重错误,当将其嵌入到一个ActivityGroup后才报错。



       android SDK中对该错误的出现的场景描述为:由于栈调用的层次太多,超过了虚拟器dalvik的最大限制(作为一个虚拟器参数,可定制)导致的。当程序编写错误导致无限递归调用时会触发,此外程序正确但是递归调用的层次过深也会触发。



       结合我的界面的情况,在单独程序中运行时没有问题,则说明不存在死循环的无限递归错误。观察该界面的集成情况,发现为了Tab页实现,该界面在显示时,被两个ActivityGroup嵌套,增加了6层显式父layout。而因为该界面复杂,设计时,list_header view的layout深度为5层,而list_item的深度则为6层。原因定位为界面嵌套层次过深。

二、解决方法
         解决嵌套过深的基本思路是在实现同样的效果的前提下减少界面的layout深度。在我将list_header view layout深度由5层精简为3层,list_item 深度由6层减少为4层后,该错误被解决。具体的方法有:
1)       多用RelativeLayout的各种布局属性来完成布局,而避免过多使用嵌套layout来实现。
2)       占位或填充剩余空间的元素多用View,避免使用layout等viewGroup。



总结下,从实际的数据来看,设计中layout嵌套深度超过10层,就应该考虑下优化了,否则就会出现java.lang.StackOverflowError的crash了。
三、错误详细日志
E/AndroidRuntime( 5196): FATAL EXCEPTION: main
E/AndroidRuntime( 5196): java.lang.StackOverflowError
E/AndroidRuntime( 5196):    at android.graphics.Paint.measureText(Paint.java:1057)
E/AndroidRuntime( 5196):    at android.text.Styled.drawDirectionalRun(Styled.java:267)
E/AndroidRuntime( 5196):    at android.text.Styled.measureText(Styled.java:430)
E/AndroidRuntime( 5196):    at android.text.Layout.measureText(Layout.java:1655)
E/AndroidRuntime( 5196):    at android.text.Layout.getLineMax(Layout.java:689)
E/AndroidRuntime( 5196):    at android.text.Layout.draw(Layout.java:340)
E/AndroidRuntime( 5196):    at android.widget.TextView.onDraw(TextView.java:4050)
E/AndroidRuntime( 5196):    at android.view.View.draw(View.java:6740)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.drawChild(ViewGroup.java:1640)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1367)
E/AndroidRuntime( 5196):    at android.view.View.draw(View.java:6743)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.drawChild(ViewGroup.java:1640)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1367)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.drawChild(ViewGroup.java:1638)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1367)
E/AndroidRuntime( 5196):    at android.view.View.draw(View.java:6743)
E/AndroidRuntime( 5196):    at android.view.View.buildDrawingCache(View.java:6502)
E/AndroidRuntime( 5196):    at android.view.View.getDrawingCache(View.java:6288)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.drawChild(ViewGroup.java:1565)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1367)
E/AndroidRuntime( 5196):    at android.widget.AbsListView.dispatchDraw(AbsListView.java:1365)
E/AndroidRuntime( 5196):    at android.widget.ListView.dispatchDraw(ListView.java:3046)
E/AndroidRuntime( 5196):    at android.view.View.draw(View.java:6846)
E/AndroidRuntime( 5196):    at android.widget.AbsListView.draw(AbsListView.java:2257)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.drawChild(ViewGroup.java:1640)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1367)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.drawChild(ViewGroup.java:1638)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1367)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.drawChild(ViewGroup.java:1638)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1367)
E/AndroidRuntime( 5196):    at android.view.View.draw(View.java:6743)
E/AndroidRuntime( 5196):    at android.widget.FrameLayout.draw(FrameLayout.java:352)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.drawChild(ViewGroup.java:1640)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1367)
E/AndroidRuntime( 5196):    at android.view.View.draw(View.java:6743)
E/AndroidRuntime( 5196):    at android.widget.FrameLayout.draw(FrameLayout.java:352)
E/AndroidRuntime( 5196):    at com.android.internal.policy.impl.PhoneWindow$DecorView.draw(PhoneWindow.java:1842)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.drawChild(ViewGroup.java:1640)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1367)
E/AndroidRuntime( 5196):    at android.view.View.draw(View.java:6743)
E/AndroidRuntime( 5196):    at android.widget.FrameLayout.draw(FrameLayout.java:352)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.drawChild(ViewGroup.java:1640)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1367)
E/AndroidRuntime( 5196):    at android.view.View.draw(View.java:6743)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.drawChild(ViewGroup.java:1640)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1367)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.drawChild(ViewGroup.java:1638)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1367)
E/AndroidRuntime( 5196):    at android.view.View.draw(View.java:6743)
E/AndroidRuntime( 5196):    at android.widget.FrameLayout.draw(FrameLayout.java:352)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.drawChild(ViewGroup.java:1640)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1367)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.drawChild(ViewGroup.java:1638)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1367)
E/AndroidRuntime( 5196):    at android.view.View.draw(View.java:6743)
E/AndroidRuntime( 5196):    at android.widget.FrameLayout.draw(FrameLayout.java:352)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.drawChild(ViewGroup.java:1640)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1367)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.drawChild(ViewGroup.java:1638)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1367)
E/AndroidRuntime( 5196):    at android.view.View.draw(View.java:6743)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.drawChild(ViewGroup.java:1640)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1367)
E/AndroidRuntime( 5196):    at android.view.View.draw(View.java:6743)
E/AndroidRuntime( 5196):    at android.widget.FrameLayout.draw(FrameLayout.java:352)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.drawChild(ViewGroup.java:1640)
E/AndroidRuntime( 5196):    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1367)
E/AndroidRuntime( 5196):    at android.view.View.draw(View.java:6743)
E/AndroidRuntime( 5196):    at android.widget.FrameLayout.draw(FrameLayout.java:352)
E/AndroidRuntime( 5196):    at com.android.internal.policy.impl.PhoneWindow$DecorView.draw(PhoneWindow.java:1842)
E/AndroidRuntime( 5196):    at android.view.ViewRoot.draw(ViewRoot.java:1407)
E/AndroidRuntime( 5196):    at
W/ActivityManager(   71):   Force finishing activity com.baidu.test/.CloudDemActivity
W/WindowManager(   71): No window to dispatch pointer action 1

Android开发中StackOverflowError的更多相关文章

  1. Android学习探索之Java 8 在Android 开发中的应用

    前言: Java 8推出已经将近2年多了,引入很多革命性变化,加入了函数式编程的特征,使基于行为的编程成为可能,同时减化了各种设计模式的实现方式,是Java有史以来最重要的更新.但是Android上, ...

  2. android开发中fragment获取context

    在用到fragment时无法使用.this来指定当前context内容,android开发中fragment获取context,可以使用getActivity().getApplicationCont ...

  3. java中的反射机制在Android开发中的用处

    JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反 ...

  4. Android开发中的输入合法性检验

    Why ? 合法性检查对于程序的健壮性具有重要作用.在Android开发中,良好的合法性检查设计机制可以使程序更加清晰,产生bug更少,交互更加友好. What ? 合法性检查的目的在于确定边界.对于 ...

  5. 在android开发中使用multdex的方法-IT蓝豹为你整理

    Android系统在安装应用时,往往需要优化Dex,而由于处理工具DexOpt对id数目的限制,导致其处理的数目不能超过65536个,因此在Android开发中,需要使用到MultiDex来解决这个问 ...

  6. 怎样实现了捕获应用中的日志在android开发中

    怎样实现了捕获应用中的日志在android开发中,大家可研究一下. Process mLogcatProc = null; BufferedReader reader = null; try { mL ...

  7. Android开发中Eclispe相关问题及相应解决(持续更新)

    1.Eclipse项目中的Android Private Libraries没有自动生成. 一般而言,在Android开发中,项目中引用到的jar包会放到项目目录中的libs中,引入库会放到Andro ...

  8. Android开发中的问题及相应解决(持续更新)

    最近博客写的少了,以后还得经常更新才行. ------------------------------------------------------------ 1.特定业务需求下try cath ...

  9. 关于Android开发中的证书和密钥等问题

    关于Android开发中的证书和密钥等问题 引言 除了Android发布应用签名时需要用到证书外,在进行google Map Api开发和Facebook SDK API开发等时都需要申请API Ke ...

随机推荐

  1. [SPOJ705]不同的子串

    题目描述] 给定一个字符串,计算其不同的子串个数. [输入格式] 一行一个仅包含大写字母的字符串,长度<=50000 [输出格式] 一行一个正整数,即不同的子串个数. [样例输入] ABABA ...

  2. Excel实现双击插入当前日期时间

    用vba做一个事件驱动程序. 第一步:点击开发工具-visual basic. 第二步:在当前工作表中编制程序如下: Private Sub Worksheet_BeforeDoubleClick(B ...

  3. Entity Framework DBContext 增删改查深度解析

    Entity Framework DBContext 增删改查深度解析 有一段时间没有更新博客了,赶上今天外面下雨,而且没人约球,打算把最近对Entity Framework DBContext使用的 ...

  4. jvm(三):对象

    关于对象,我们需要面对的问题主要有对象的创建,对象在内存中的布局,对象的结构,对象的访问定位. 对象的创建 对象的创建过程如下图所示: 其主要步骤有:给对象分配内存,初始化对象,执行构造方法. 在对象 ...

  5. zkCli的使用 常用的节点增删改查命令用法

    zkCli的使用 常用的节点增删改查命令用法 1. 建立会话  命令格式:zkCli.sh -timeout 0 -r -server ip:port ./zkCli.sh -server -time ...

  6. JVM常见问题(二)

    6. GC收集器有哪些?它们的特点是? 常见的GC收集器如下图所示,连线代表可搭配使用: 1.Serial收集器(串行收集器) 用于新生代的单线程收集器,收集时需要暂停所有工作线程(Stop the ...

  7. tree的遍历--广度优先遍历

    一.二叉树demo var tree = { value: '一', left: { value: '二', left: { value: '四', right: { value: '六' } } } ...

  8. manjaro备忘录

    updated 2018/4/3 manjaro 使用Linux发行版时需要注意几个方面的问题: 包管理器 包管理器无疑时各家发行版的最大特色之一.软件同时也是一个平台是否能够产生足够的吸引力的来源之 ...

  9. HTTP与TCP的关系

    一直比较想写TCP与HTTP之间的关系,HTTP报文是如何通过tcp发送的,HTTP报文形式内容如何. HTTP请求包含请求行,请求头,请求体 HTTP响应包含响应头,响应头,响应体 下面我准备通过J ...

  10. Template基础

    模板系统的介绍 你可能已经注意到我们在例子视图中返回文本的方式有点特别. 也就是说,HTML被直接硬编码在 Python代码之中. def current_datetime(request): now ...