问题分析
RuntimeException是运行时异常,是Java编译器事先不可预见的异常。RuntimeException的子类众多,这里列举一些比较常见的子类:
BufferOverflowException当相关put操作达到目标缓冲区限制时,抛出此未经检查的异常。
ArrayStoreException试图将错误类型的对象存储到一个对象数组时抛出的异常。
ArithmeticException当出现异常的运算条件时,抛出此异常。例如,一个整数“除以零”时,抛出此类的一个实例。
BufferUnderflowException当相关get操作达到源缓冲区限制时,抛出此未经检查的异常。
IndexOutOfBoundsException指示某排序索引(例如对数组、字符串或向量的排序)引用时超出范围时抛出。
NoSuchElementException由Enumeration的nextElement方法抛出,表明枚举中没有更多的元素。
解决方案
RuntimeException是发生在程序运行期,预先不可预见的发生。编译器未要求一定要进行捕获,如果运行期没有处理,则RuntimeException会一直往上层抛。最后由JVM来处理,JVM会打印堆栈信息然后结束应用。对于可能发生的RuntimeException,建议根据堆栈信息,检查代码是否有误并进行更改,如果情况复杂无法全部解决,可以对RuntimeException进行捕获并进行业务恢复处理。
代码示例
try {
somethingThrowingARuntimeException();
}catch (RuntimeException e) {
// Do something with it. At least log it
e.printStackTrace();
}
示例一
java.lang.RuntimeException:Unable to start activity ComponentInfo{*}: android.support.v4.app.Fragment$InstantiationException:*: make sure class name exists,ispublic,and has an empty constructor that ispublic
当系统因为内存不足杀死非前台进程,用户又将被系统杀掉的非前台应用带回前台,如果这个时候有UI是呈现在Fragment中,那么会因为restore造成fragment需要通过反射实例对象,从而将之前save的状态还原。因此不应该改写构造函数,而应该使用静态方法比如newInstance来构造有参Fragment对象并返回。
代码示例:
步骤一:编写有参静态创建对象函数
public static final AlertFragment newInstance(int title, String message)
{
AlertFragment f = new AlertFragment();
Bundle bdl = new Bundle(2);
bdl.putInt(EXTRA_TITLE, title);
bdl.putString(EXTRA_MESSAGE, message);
f.setArguments(bdl);
return f;
}
步骤二:通过Bundle获取参数
@Override
public void onCreate(Bundle savedInstanceState)
{
title = getArguments().getInt(EXTRA_TITLE);
message = getArguments().getString(EXTRA_MESSAGE);
//...
//etc
//...
}
步骤三:通过FragmentManager对Fragment进行管理。
public onCreate(Bundle savedInstanceState) {
if(savedInstanceState == null){
getSupportFragmentManager()
.beginTransaction()
.replace(R.id.content,AlertFragment.newInstance(
R.string.alert_title,
"Oh noes an error occured!")
)
.commit();
}
}
示例二
Exception in thread "main" java.lang.IndexOutOfBoundsException: Index: 80, Size: 3
at java.util.ArrayList.rangeCheck(ArrayList.java:635)
at java.util.ArrayList.remove(ArrayList.java:474)
at com.alibaba.mqc.test.Test.indexOutOfBounds(Test.java:66)
at com.alibaba.mqc.test.Test.main(Test.java:32)
问题分析:
IndexOutOfBoundsException 指示某排序索引(例如对数组、字符串或向量的排序)引用时超出范围时抛出。请检查是否有对List有越界索引,判断访问数组的索引是否合法。
有误的代码:
public void indexOutOfBounds(){
ArrayList<Character> a = new ArrayList<Character>();
a.add('A');
a.add('B');
a.add('C');
System.out.println(a);
//a.remove('A');
a.remove(80); // 抛出此类异常
System.out.println(a);
}
建议的代码:
public void indexOutOfBounds(){
ArrayList<Character> a = new ArrayList<Character>();
a.add('A');
a.add('B');
a.add('C');
System.out.println(a);
if(a.size()>80){
a.remove(80);
}
System.out.println(a);
}
参考文献
https://developer.android.com/reference/java/lang/RuntimeException.html
http://stackoverflow.com/questions/10450348/do-fragments-really-need-an-empty-constructor
http://stackoverflow.com/questions/2028719/handling-runtimeexceptions-in-java
http://tool.oschina.net/uploads/apidocs/jdk-zh/java/lang/RuntimeException.html