Android常见异常处理

什么是异常?

Exception是指在程序运行过程中所出现的不可预测的可处理可恢复的问题,这些错误会干扰到指令的正常执行,从而造成程序的异常退出或者意外终止。比如:FileNotFountException、NullPointException.

Error是指程序运行过程中较为严重的不可挽回的问题,大多数错误与代码编写者执行的操作无关,而表示代码运行时 JVM(Java 虚拟机)出现的问题。比如OutOfMemeryError,一旦出现错误程序就要挂。

常见异常分类

编译时错误(比如:ClassNotFoundException)

从程序语法角度讲是必须进行处理的异常,如果不处理,程序就不能编译通过。如IOException、SQLException等以及用户自定义的Exception异常

运行时错误(比如:ArrayIndexOutOfBoundException)

这些异常是不检查异常,程序中可以选择捕获处理,也可以不处理。这些异常一般是由程序逻辑错误引起的,程序应该从逻辑角度尽可能避免这类异常的发生。
1、 java.lang.ArrayIndexOutOfBoundsException
数组索引越界异常。当对数组的索引值为负数或大于等于数组大小时抛出。
2、java.lang.ArithmeticException
算术条件异常。譬如:整数除零等。
3、java.lang.NullPointerException
空指针异常。当应用试图在要求使用对象的地方使用了null时,抛出该异常。譬如:调用null对象的实例方法、访问null对象的属性、计算null对象的长度、使用throw语句抛出null等等
4、java.lang.ClassNotFoundException
找不到类异常。当应用试图根据字符串形式的类名构造类,而在遍历CLASSPAH之后找不到对应名称的class文件时,抛出该异常。
5、java.lang.NegativeArraySizeException 数组长度为负异常
6、java.lang.ArrayStoreException 数组中包含不兼容的值抛出的异常
7、java.lang.SecurityException 安全性异常
8、java.lang.IllegalArgumentException 非法参数异常

异常处理机制

在 Java 应用程序中,异常处理机制为:抛出异常,捕捉异常。

Java中防止空指针策略

1.使用模板方法模式规范生命周期(次序)

1
2
3
4
5
6
7
8
9
10
11
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(getContentView());
initView();
mPresenter = initPresenter();
if(mPresenter != null) {
mPresenter.attachView(this);
}
initData();
}

2.明确需要创建的对象优先new出来

1
private List<Subscription> mSubscriptions = new ArrayList<>();

3.模块内检查非成员变量是否非空(不可靠检查)

1
2
3
4
public void setUserData(User user){
if(user == null) return;
//TODO
}

4.return 对象 可能为空写上必要注释

1
2
3
4
5
6
7
8
/**
* 获取用户数据
* @return User 可能返回NULL
*/
public User getUserData(){

return null;
}

多线程异常

http://blog.csdn.net/u013256816/article/details/50417822

Thread的run方法是不抛出任何检查型异常(checked exception)的,但是它自身却可能因为一个异常而被终止,导致这个线程的终结。最麻烦的是,在线程中抛出的异常即使使用try…catch也无法截获,因此可能导致一些问题出现,比如异常的时候无法回收一些系统资源,或者没有关闭当前的连接等等。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
package com.example.uncaughtException;

/**
* Created by 水寒 on 2017/11/6.
*/

public class NoCaughtThread {

public static void main(String[] args){
try {
Thread thread = new Thread(new Task());
thread.start();
}catch(Exception e){
System.out.println("Exception:" + e.getMessage());
}

System.out.println("Main Thread Started....");

try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}

System.out.println("Main Thread Sleep 1000....");
}
}

class Task implements Runnable{

@Override
public void run() {
System.out.println(3 / 2);
System.out.println(3 / 0);
System.out.println(3 / 1);
}
}

JDK5.0之前,不能为单独的Thread设置UncaughtExceptionHandler,也不能指定一个默认的UncaughtExceptionHandler。为了可以设置一个UncaughtExceptionHandler,需要去继承ThreadGroup并覆写uncaughtException方法。

在JDK5.0中,我们通过Thread的实例方法setUncaughtExceptionHandler,可以为任何一个Thread设置一个UncaughtExceptionHandler。当然你也可以为所有Thread设置一个默认的UncaughtExceptionHandler,通过调用Thread.setDefaultUncaughtExceptionHandler(Thread.UncaughtExceptionHandler eh)方法,这是Thread的一个static方法。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public class NoCaughtThread {

public static void main(String[] args){
//Thread.setDefaultUncaughtExceptionHandler(new ExceptionHandler());
Thread thread = new Thread(new Task());
thread.setUncaughtExceptionHandler(new ExceptionHandler());
thread.start();
}
}

class ExceptionHandler implements Thread.UncaughtExceptionHandler {

@Override
public void uncaughtException(Thread thread, Throwable throwable) {
System.out.println(thread.getName() + " Throw : " + throwable.getMessage());
}
}

Android定制错误日志系统

每当我们app测试的时候,测试人员总是对我们说这里崩溃了,那里挂掉了!我们只能默默接受,然后尝试着重现bug,更可悲的是有时候bug很难复现,为了解决这种现状所以我们要尝试这建立一个自己的bug日志系统。

Java为我们提供了一个机制,用来捕获并处理在一个线程对象中抛出的未检测异常,以避免程序终止。我们可以通过UncaughtExceptionHandler来实现这种机制。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
package guohe.testkotlin.manager;

import android.app.Application;
import android.content.Context;

import java.io.PrintWriter;
import java.io.StringWriter;
import java.io.Writer;

/**
* Created by 水寒 on 2017/11/6.
* 奔溃日志管理
*/

public class CrashManager implements Thread.UncaughtExceptionHandler{

private static CrashManager mInstance;
private Application mApplication;
// 系统默认的UncaughtException处理类
private Thread.UncaughtExceptionHandler mDefaultHandler;

private CrashManager(Context context){
mApplication = (Application) context.getApplicationContext();
mDefaultHandler = Thread.getDefaultUncaughtExceptionHandler();
//替换handler, 这是在主线程里执行
Thread.setDefaultUncaughtExceptionHandler(this);
}

public static CrashManager getInstance(Context context){
if(mInstance == null){
mInstance = new CrashManager(context);
}
return mInstance;
}

@Override
public void uncaughtException(Thread t, Throwable e) {
printLog(t, e);
mDefaultHandler.uncaughtException(t, e);
}

private void printLog(Thread t, Throwable e){
Writer writer = new StringWriter();
PrintWriter printWriter = new PrintWriter(writer);
e.printStackTrace(printWriter);
Throwable cause = e.getCause();
while (cause != null) {
cause.printStackTrace(printWriter);
cause = cause.getCause();
}
printWriter.close();
String result = writer.toString(); //这就是异常日志堆栈信息
System.out.println("Thread:" + t.getName() + ", exception: " + result);
}
}

Android Crash处理GitHub开源库

https://github.com/ACRA/acra

https://github.com/Sunzxyong/Recovery

https://github.com/drakeet/CrashWoodpecker