《Android应用开发攻略》——2.2 异常处理

简介: 本节书摘来自华章计算机《Android应用开发攻略》一书中的第2章,第2.2节,作者:(美)达尔文(Darwin, I. F.)著, 更多章节内容可以访问云栖社区“华章计算机”公众号查看。

2.2 异常处理

Ian Darwin
2.2.1 问题
Java有一个精心定义的异常处理机制,但是需要花费一定的时间学习,才能高效地使用而不至于使用户或者技术支持人员感到沮丧。
2.2.2 解决方案
Java提供了一个Exception层次结构,正确地使用它能够带来相当大的灵活性。Android提供了包括对话框和Toast的多种机制,用于通知用户错误的情况。Android开发者应该熟悉这些机制,并学习高效使用它们的方法。
2.2.3 讨论
Java从推出时就有两类异常(实际上是Exception类的父类Throwable):检测型(checked)异常和非检测型(unchecked)异常。在Java标准版中,编程人员明显要面对这样一个事实:在编译的时候可以检测到某些情况,但其他的情况则无法检测。例如,如果许多PC上安装一个桌面应用程序,可能有些PC上的磁盘空间已经很紧张,无法保存数据;与此同时,其他PC上应用程序依赖的一些文件可能由于用户的错误(而非编程人员的错误)、偶发事件、老鼠咬断电缆等情况而丢失。因此,IOException被当作“检查型异常”,意味着编程人员必须检查这类异常,检查可以通过文件使用方法中的try-catch子句或者方法定义中的throws子句来完成。所有经过良好训练的Java开发人员都知道如下的通用规则:
Throwble是可抛出异常层次结构的根。Exception及其子类(RuntimeException(及其子类)除外)都是检测型异常。其他异常为非检测型异常。
上述规则意味着,Error及其所有子类都是非检测异常(见图2-1)。例如,如果你收到一个VMError异常,说明出现了一个运行时bug,作为应用程序编程人员,对此你没有什么可做的。RuntimeException子类包含了名称超长的ArrayIndexOutOfBoundsException等异常,它和它的友元都是非检测型异常,因为在开发时测试及捕捉这些异常是你的责任(参见第3章)。

image


捕捉异常的场所
早期对检测型异常的(过度)使用导致许多早期Java开发人员编写的代码中到处都是try/catch代码块,部分原因是:在早期的一些培训项目和书籍中,对throws子句的使用没有得到足够的重视。随着Java本身越来越多地转向企业应用,较新的框架(如Hibernate和Spring)出现并强调非检测型异常的使用,这类问题才得到改正。现在,尽可能捕捉靠近用户的异常这一理念已经得到普遍的接受。准备(在程序库或者多个应用程序中)重用的代码不应该尝试错误处理,它们所能做的是所谓“异常转译”(Exception Translation),也就是说,将与技术相关的(通常是检查型)异常转换为通用的非检测型异常。例2-1展示了基本的模式。
例2-1:异常转译
public String readTheFile(String f) {
        BufferedReader is = null;
        try {
                is = new BufferedReader(new FileReader(f));
                String line = is.readLine();
                return line;
       } catch (FileNotFoundException fnf) {
               throw new RuntimeException("Could not open file " + f, fnf);
       } catch (IOException ex) {
               throw new RuntimeException("Could not read file " + f, ex);
       } finally {
              if (is != null) {
                      try {
                              is.close();
                      } catch(IOException grr) {
                              throw new RuntimeException("Error on close of " + f, grr);
                      }
              }
       }
}

注意,即便在这个代码中,检测型异常的用法也很零乱:is.close()实际上不可能失败,但是因为将它放在finally块中(如果文件打开但是出现了某种错误,这样可以确保文件关闭),就必须使用另一个try-catch结构。因此,检测型异常(很可能)是不好的,在新的API中应该避免使用它,在必要时应该用非检测型异常来扫清道路。
Oracle官方网站和其他人则支持相反的看法。在本书网站上的一条评论中,Al Sutton提出了如下看法:
检测型异常迫使开发人员承认错误的情况可能发生,他们必须思考处理错误的方式。在许多情况下,除了日志和恢复可能没有太多的事情可做,但是开发人员仍然考虑到此类错误发生的时候所应采取的措施。这个例子说明阻止方法调用者区分文件不存在(因而没有必要重新读取)和文件读取问题(文件存在但是无法读取)这两种不同错误情况之间的区别。
Android忠于JavaAPI,有许多检测型异常(包括例子中说明的那些),因此对这些异常应该以相同的方式处理。
对异常的处理
应用程序应该始终报告异常。当我看到捕捉的异常却不采取任何措施的代码时,我总会感到绝望。但是,异常只应该报告一次(不要既进行记录,又转译/重新抛出异常)。普通异常的重点是表示异常情况(正如名称所表示的那样)。因为在Android设备上没有系统管理员或者控制台操作员,所以必须向用户报告异常情况。
你应该考虑通过一个对话框或者Toast通知报告异常。移动设备上的异常处理与台式机有所不同。用户可能正在开车(或者操作其他机器),与人交谈,因此你不应该猜测他们的注意力都在应用程序上。记住,Toast通知只在屏幕上出现几秒钟,你可能会错过它。如果用户必须采取行动更正错误,就应该使用对话框。我知道大部分的示例(甚至在本书中也是一样)使用Toast,这是因为Toast通知需要的代码比对话框少(相反,BlackBerry API简化了对话框:Dialog.alert("mes sage"))。Toast简单地弹出,然后消失。对话框则要求用户确认异常情况,或者授权应用程序执行某些付费操作(例如,启动互联网访问,以便运行需要下载地图数据的应用程序)。
注意: 使用Toast来“弹出”不重要的信息,而使用对话框显示重要的信息并得到确认。

相关文章
|
10天前
|
IDE Java 开发工具
深入探索安卓应用开发:从环境搭建到第一个"Hello, World!"应用
本文将引导读者完成安卓应用开发的初步入门,包括安装必要的开发工具、配置开发环境、创建第一个简单的安卓项目,以及解释其背后的一些基本概念。通过一步步的指导和解释,本文旨在为安卓开发新手提供一个清晰、易懂的起点,帮助读者顺利地迈出安卓开发的第一步。
188 65
|
10天前
|
存储 Java Android开发
探索安卓应用开发:构建你的第一个"Hello World"应用
【9月更文挑战第24天】在本文中,我们将踏上一段激动人心的旅程,深入安卓应用开发的奥秘。通过一个简单而经典的“Hello World”项目,我们将解锁安卓应用开发的基础概念和步骤。无论你是编程新手还是希望扩展技能的老手,这篇文章都将为你提供一次实操体验。从搭建开发环境到运行你的应用,每一步都清晰易懂,确保你能顺利地迈出安卓开发的第一步。让我们开始吧,探索如何将一行简单的代码转变为一个功能齐全的安卓应用!
|
15天前
|
存储 Oracle Java
深入探索安卓应用开发:从环境搭建到第一个"Hello, World!"
本文旨在为安卓开发初学者提供一个清晰、简洁的入门指南。我们将一步步引导您完成安卓开发环境的搭建,并详细介绍如何创建您的第一个安卓应用程序。通过这篇文章,您不仅能够理解安卓应用开发的基本流程,还能掌握一些实用的技巧和方法,为进一步深入学习打下坚实的基础。
|
3天前
|
存储 测试技术 Android开发
探索安卓应用开发:从基础到高级
【9月更文挑战第31天】在这篇文章中,我们将一起踏上安卓应用开发的旅程。无论你是初学者还是有一定经验的开发者,本文都将为你提供有价值的信息和指导。我们将从安卓应用开发的基础知识开始,逐步深入到更高级的主题。通过阅读本文,你将了解到如何构建一个安卓应用,包括用户界面设计、数据存储和网络通信等方面。此外,我们还将探讨一些高级主题,如性能优化、安全性和测试。让我们一起开始吧!
|
4天前
|
设计模式 Java Android开发
安卓应用开发中的内存泄漏检测与修复
【9月更文挑战第30天】在安卓应用开发过程中,内存泄漏是一个常见而又棘手的问题。它不仅会导致应用运行缓慢,还可能引发应用崩溃,严重影响用户体验。本文将深入探讨如何检测和修复内存泄漏,以提升应用性能和稳定性。我们将通过一个具体的代码示例,展示如何使用Android Studio的Memory Profiler工具来定位内存泄漏,并介绍几种常见的内存泄漏场景及其解决方案。无论你是初学者还是有经验的开发者,这篇文章都将为你提供实用的技巧和方法,帮助你打造更优质的安卓应用。
|
6天前
|
缓存 前端开发 Android开发
安卓应用开发中的自定义控件
【9月更文挑战第28天】在安卓应用开发中,自定义控件是提升用户界面和交互体验的关键。本文通过介绍如何从零开始构建一个自定义控件,旨在帮助开发者理解并掌握自定义控件的创建过程。内容将涵盖设计思路、实现方法以及性能优化,确保开发者能够有效地集成或扩展现有控件功能,打造独特且高效的用户界面。
|
9天前
|
存储 XML 前端开发
探索Android应用开发:从基础到进阶
【8月更文挑战第57天】在这篇文章中,我们将深入探讨Android应用开发的奥秘。无论你是新手还是有经验的开发者,本文都将为你提供有价值的见解和技巧。我们将从基本的UI设计开始,逐步介绍数据存储、网络请求等高级主题,并展示一些实用的代码示例。让我们一起踏上这段激动人心的旅程吧!
|
15天前
|
开发工具 Android开发 iOS开发
掌握安卓与iOS应用开发:关键技术与未来展望
本文深入探讨了安卓和iOS平台下的应用开发技术,重点比较了两大平台的架构、开发工具和市场策略。通过分析最新的技术趋势和开发者社区的反馈,文章为读者提供了一个全面的对比视角,旨在帮助开发者做出更明智的平台选择和开发决策。
|
10天前
|
安全 Java Android开发
掌握安卓与iOS应用开发中的关键技术
本文深入探讨了安卓和iOS平台上应用开发的关键性技术,包括平台特性、开发工具选择、性能优化技巧及跨平台开发的可行性分析。通过对比两种平台的开发环境与实践案例,旨在为开发者提供全面的视角以理解和把握移动应用开发的核心技术。无论是安卓的Java与Kotlin之争,还是iOS的Swift语言革命,本文都将一一解析其优势与应用场景,帮助开发者在技术选型上有更明智的决策。此外,文章还将触及到当前流行的跨平台框架如React Native和Flutter,评估它们在项目实施中的实用性和限制,为有意进行多平台同步开发的团队提供参考。通过对这些关键技术的梳理,本文期望能够启发开发者深化对移动平台开发的理解,并
|
12天前
|
存储 开发框架 数据可视化
深入解析Android应用开发中的四大核心组件
本文将探讨Android开发中的四大核心组件——Activity、Service、BroadcastReceiver和ContentProvider。我们将深入了解每个组件的定义、作用、使用方法及它们之间的交互方式,以帮助开发者更好地理解和应用这些组件,提升Android应用开发的能力和效率。
下一篇
无影云桌面