[android]在 TreeMap.containsKey() 中的怪诞时抛出

标签: json Java Android gson
发布时间: 2017/3/4 2:02:52
注意事项: 本文中文内容可能为机器翻译,如要查看英文原文请点击上面连接.

这是在哪里崩溃出现的行

offsetDuration = duration - (offsets.containsKey(freq) 
                    ? offsets.get(freq) : 0l);

我通过捕获的值 Exception 和倾倒的变量,

long offsetDuration = 0;
long duration = 391144;
TreeMap<Long, Long> offsets = {0=4024974.0, 1036800=8588.0, 1190400=88216.0, 1267200=49763.0, 1497600=87476.0, 1574400=7469.0, 1728000=54553.0, 1958400=60512.0, 2265600=246942.0, 300000=390779.0, 422400=39945.0, 652800=55204.0, 729600=46829.0, 883200=19191.0, 960000=23888.0}
long freq = 300000;

该变量 TreeMap<Long, Long> offsets 通过使用下面的代码从 json 文件解析。

@NonNull
public static TreeMap<Long, Long>  getOffsets(Context context) throws CpuStateException {
    File file = getOffsetsFile(context);
    TreeMap<Long, Long> map;

    try {
        String s = Files.toString(file, Charsets.UTF_8).trim();
        Gson gson = new GsonBuilder().create();
        Type type = new TypeToken<TreeMap<Long, Long>>(){}.getType();
        map = gson.fromJson(s, type);
    } catch (IOException e) {
        throw new CpuStateException("Failed to read offsets!");
    }

    if (map == null)
        throw new CpuStateException("Failed to read offsets!");

    return map;
}

后检查代码很多次,我不能识别此代码可以扔在那里的情况

java.lang.ClassCastException: java.lang.String cannot be cast to java.lang.Long
    at java.lang.Long.compareTo(Long.java:32)
    at java.util.TreeMap.find(TreeMap.java:277)
    at java.util.TreeMap.findByObject(TreeMap.java:351)
    at java.util.TreeMap.containsKey(TreeMap.java:182)
    at com.vibhinna.library.engine.CpuStates.getCpuData(CpuStates.java:96)
    at com.vibhinna.library.engine.CpuStates.getBarData(CpuStates.java:162)

有什么想法?

更新 1︰ 这是 json 如何生成︰

public static void offsetTimers(TreeMap<Long, Long> offsets, Context context) throws CpuStateException {
    Gson gson = new GsonBuilder().create();
    String json = gson.toJson(offsets);
    File file = getOffsetsFile(context);

    try {
        OutputStreamWriter outputStream = new OutputStreamWriter(new FileOutputStream(file),
                "UTF-8");
        outputStream.write(json);
        outputStream.flush();
        outputStream.close();
    } catch (IOException e) {
        throw new CpuStateException("Failed to save offsets!");
    }
}

更新 2 ︰ 我不是甚至能够重现这甚至在原应用程序中,飞机坠毁是罕见的报告由关于只有在小于 1%的客户。

更新 3

offset class: class java.lang.String    value: 1036800
offset class: class java.lang.String    value: 1190400
offset class: class java.lang.String    value: 1267200
offset class: class java.lang.String    value: 1497600
offset class: class java.lang.String    value: 1574400
offset class: class java.lang.String    value: 1728000
offset class: class java.lang.String    value: 1958400
offset class: class java.lang.String    value: 2265600
offset class: class java.lang.String    value: 300000
offset class: class java.lang.String    value: 422400
offset class: class java.lang.String    value: 652800
offset class: class java.lang.String    value: 729600
offset class: class java.lang.String    value: 883200
offset class: class java.lang.String    value: 960000

json:

{  
   "0":256093,
   "300000":105045,
   "422400":9677,
   "652800":10443,
   "729600":8868,
   "883200":3951,
   "960000":7323,
   "1036800":18668,
   "1190400":34938,
   "1267200":17151,
   "1497600":11018,
   "1574400":1173,
   "1728000":22881,
   "1958400":21076,
   "2265600":66501
}

解决方法 1:

这里是丑陋的解决方法,似乎工作。

@NonNull
public static TreeMap<Long, Long>  getOffsets(Context context) throws CpuStateException {
    File file = getOffsetsFile(context);
    TreeMap map;

    try {
        String s = Files.toString(file, Charsets.UTF_8).trim();
        Gson gson = new GsonBuilder().create();
        Type type = new TypeToken<TreeMap<Long, Long>>(){}.getType();
        map = gson.fromJson(s, type);
    } catch (IOException e) {
        throw new CpuStateException("Failed to read offsets!");
    }

    if (map == null)
        throw new CpuStateException("Failed to read offsets!");

    // So, I don't trust Gson. On some devices the above code seems to be producing a
    // TreeMap<String, Double/Float>. So let's assume it's either a String or a boxed primitive.
    // Even if the assumption is wrong we're no worse off than we were before.
    // The String we'll parse, the primitive we'll cast. Brace yourselves for a horrible hack!
    // TODO find a proper fix.
    Iterator it = map.keySet().iterator();
    TreeMap<Long, Long> treeMap = new TreeMap<>();
    while (it.hasNext()) {
        Object key = it.next();
        Object value = map.get(key);
        Long lKey, lValue;

        if ((key instanceof Long)) {
            lKey = (Long) key;
        } else {
            Log.e(TAG, "Invalid key type in TreeMap: " + key.getClass());
            if (key instanceof Number) {
                lKey = ((Number) key).longValue();
            } else if (key instanceof String) {
                lKey = Double.valueOf((String) key).longValue();
            } else {
                throw new IllegalArgumentException("Invalid key in TreeMap: "  + key);
            }
        }

        if ((value instanceof Long)) {
            lValue = (Long) value;
        } else {
            Log.e(TAG, "Invalid value type in TreeMap: " + value.getClass());
            if (value instanceof Number) {
                lValue = ((Number) value).longValue();
            } else if (value instanceof String) {
                lValue = Double.valueOf((String) value).longValue();
            } else {
                throw new IllegalArgumentException("Invalid value in TreeMap: " + value);
            }
        }
        treeMap.put(lKey, lValue);
    }

    return treeMap;
}
官方微信
官方QQ群
31647020