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. ●BZOJ 2669 [cqoi2012]局部极小值

    题链: http://www.lydsy.com/JudgeOnline/problem.php?id=2669 题解: 容斥,DP,DFS 先看看 dp 部分:首先呢,X的个数不会超过 8个.个数很 ...

  2. [Codeforces]605E Intergalaxy Trips

    小C比较棘手的概率期望题,感觉以后这样的题还会贴几道出来. Description 给定一个n*n的邻接矩阵,邻接矩阵中元素pi,j表示的是从 i 到 j 这条单向道路在这一秒出现的概率百分比,走一条 ...

  3. 数据结构与算法 —— 链表linked list(05)

    反转一个单链表. 进阶:链表可以迭代或递归地反转.你能否两个都实现一遍? 示例 : 给定这个链表:1->2->3->4->5 返回结果: 5->4->3->2 ...

  4. MFC 程序入口和执行流程

    MFC(微软基础类库)以C++类的形式封装了Windows API,给开发者提供了便利,但是初学者常常会疑惑MFC程序的入口在哪里?下面给大家简单介绍一下MFC 程序入口和执行流程. 一 MFC程序执 ...

  5. 正则替换replace中$1的用法以及常用正则

    一.repalce定义 用于在字符串中用一些字符替换另一些字符,或替换一个与正则表达式匹配的子串. stringObject.replace(regexp/substr,replacement)参数一 ...

  6. SVN冲突解决

    问题一.执行SVN commit时候,会生成除正常文件之外.mine..r3439 ..r3368的三个文件 .mine:是自己要提交的版本 .r3439:在别人之前提交的版本 .r3368:初始版本 ...

  7. Oracle中建库时报Exception in thread main

    Linux操作系统上安装oracle 10g,在启动dbca的时候报 Exception in thread "main" 错误,详细内容如下: [oracle@centos ~] ...

  8. PHP 5 String 函数

    PHP 5 String 函数 PHP String 函数是 PHP 核心的组成部分.无需安装即可使用这些函数. 函数 描述 addcslashes() 返回在指定的字符前添加反斜杠的字符串. add ...

  9. 集群技术(二) MySQL集群简介与配置详解

    when?why? 用MySQL集群? 减少数据中心结点压力和大数据量处理(读写分离),采用把MySQL分布,一个或多个application对应一个MySQL数据库.把几个MySQL数据库公用的数据 ...

  10. 设子数组A[0:k]和A[k+1:N-1]已排好序(0≤K≤N-1)。试设计一个合并这2个子数组为排好序的数组A[0:N-1]的算法。

    设子数组A[0:k]和A[k+1:N-1]已排好序(0≤K≤N-1).试设计一个合并这2个子数组为排好序的数组A[0:N-1]的算法.要求算法在最坏情况下所用的计算时间为O(N),只用到O(1)的辅助 ...