发生IndexOutOfBoundsException错误取决于Android操作系统版本

时间:2013-03-11 12:58:54

标签: android json indexoutofboundsexception

我正在拼命地尝试修复一个错误:

  • 总是发生在Android版本2.2,2.3的模拟器中
  • 永远不会出现在模拟器Android版本4。*
  • 从未在真实设备中发生过(android 4。*)

以下是IndexOutOfBoundsException异常:

java.lang.RuntimeException: Unable to start activity
ComponentInfo{<myapppackage>}: java.lang.IndexOutOfBoundsException:
Invalid index 39, size is 0

在我的应用程序中,我正在显示我正在显示为文本的json文件中的数据。我已经将错误发送到了哪里,就在我调用这种方法的时候:

public String getItemValue(int id, String s) {      

    List<JsonItems> list = new ArrayList<JsonItems>();

    try {
        // CONVERT RESPONSE STRING TO JSON ARRAY
        JSONArray ja = new JSONArray(s);

        // ITERATE THROUGH AND RETRIEVE 
        int n = ja.length();            

        for (int i = 0; i < n; i++) {
            // GET INDIVIDUAL JSON OBJECT FROM JSON ARRAY
            JSONObject jo = ja.getJSONObject(i);

            // RETRIEVE EACH JSON OBJECT'S FIELDS
            JsonItems ji = new JsonItems();
            ji.id = jo.getInt("id");
            ji.text= jo.getString("text");
            list.add(ji);

        }
    } catch (Exception e) {
        e.printStackTrace();
    }
    return list.get(id).text; 
}

我的类JsonItems非常基础:

public class JsonItems{
int id;
String text;
}

来自我的json文件的示例:

[
{"id":0,"text":"some text 0"},
{"id":1,"text":"some text 1"},
{"id":2,"text":"some text 2"}
]

以下是我将json文件的内容处理为String

的方法
public static String fromJsonFileToString(String fileName, Context c) {
    //JSONArray jArray = null;
    String text = "";
    try {
        InputStream is = c.getAssets().open(fileName);
        int size = is.available();
        byte[] buffer = new byte[size];
        is.read(buffer);
        is.close();
        text = new String(buffer);

    } catch (IOException e) {
        throw new RuntimeException(e);
    }
    return text;

}

我再次重申:IndexOutOfBoundsException永远不会发生在使用Android 4. *的设备上,只有在我使用Android 2在模拟器上测试应用时才会发生。*

知道它来自何处?

由于

1 个答案:

答案 0 :(得分:2)

第一个问题

您没有正确阅读输入流。从字面上调用available就是这样 - 它会返回可用要读取的数据量, 进行调用时 此数字可能或可能不代表文件的整个内容

为您阅读材料:

请注意,像Apache Commons IO这样的辅助库可以在一行代码(IOUtils.toString(inputStream))中读取文件内容。 Android不支持Java 7,但在该版本中使用Files.readAllLines方法提供了一个值得注意的替代方案。在任何情况下,您都可以对文件读取代码进行以下显示的更改,它应该更好:

public static String fromFileToString(String fileName, Context context) {
    try {
        BufferedReader reader = new BufferedReader(new InputStreamReader(
                context.getAssets().open(fileName)));

        String line = null;
        StringBuilder builder = new StringBuilder(1024);
        while ((line = reader.readLine()) != null) {
            builder.append(line);
        }

        return builder.toString();
    } catch (IOException e) {
        throw new RuntimeException(e);
    }
}

第二个问题:

您没有进行任何绑定检查,以确保您传递给“搜索”方法的参数:

public String getItemValue(int id, String s)

不超过您最终计算的项目列表的长度:

return list.get(id).text;
//              ^
// -------------
// 'id' could be larger than the list size!

在任何情况下,您当前的设计根本不符合您真正尝试做的事情,也就是说,确定JSON数组中具有匹配的“id”字段的元素你提供给方法。您需要将JSON数据作为地图处理,以便能够执行此操作。

public String getItemValue(int id, String json) {
    if(json == null || json.trim().equals("")) return null;

    Map<Integer, JsonItems> map = new HashMap<Integer, JsonItems>(4);

    try {
        JSONArray ja = new JSONArray(json);

        int n = ja.length();

        for (int i = 0; i < n; i++) {
            JSONObject jo = ja.getJSONObject(i);

            JsonItems ji = new JsonItems();
            ji.id = jo.getInt("id");
            ji.text = jo.getString("text");
            map.put(Integer.valueOf(ji.id, ji);
        }
    } catch(NumberFormatException nfe) {
      throw new IllegalArgumentException("Invalid JSON format");  
    } catch (Exception e) {
        throw new RuntimeException(e);
    }

    JsonItems item = map.get(id);
    return item != null ? item.text : null;
}

一些快速说明:

  • JsonItems应该被称为JsonItem,以符合良好的Java命名标准
  • 您应该只解析并存储一次JSON,以提高性能
  • 您实际上只使用JSON数据的最小子集,您实际上可以确定for循环中的匹配节点并直接返回其值,而不必使用intermedia Java bean对象