部分OPPO机型 AssetManager.finalize() timed out的修复

在线上的APP常常收到以下的崩溃报告java

.finalize() timed outandroid

image.png

在排除了内存泄漏的状况后,仔细核查后发现多发在OPPO系列的机型中,包括R9 A33 A59等等。猜想是OPPO的定制ROM在底层作了什么修改。为了减小这样的报错,只有对OPPO进行特殊处理。 查找了一些资料后发现这类错误是因为回收对象时间过长,由FinalizerWatchdogDaemon负责计时,超时后抛出异常关闭VM的。 那么有两种解决办法:bash

1关掉这个负责计时的,2延长计时时间ide

try {
    Class clazz = Class.forName("java.lang.Daemons$FinalizerWatchdogDaemon");
    Method method = clazz.getSuperclass().getDeclaredMethod("stop");
    method.setAccessible(true);
    Field field = clazz.getDeclaredField("INSTANCE");
    field.setAccessible(true);
    method.invoke(field.get(null));
} catch (Throwable e) {
    e.printStackTrace();
}
复制代码

或者字体

try {
    Class<?> c = Class.forName("java.lang.Daemons");
    Field maxField = c.getDeclaredField("MAX_FINALIZE_NANOS");
    maxField.setAccessible(true);
    maxField.set(null, Long.MAX_VALUE);
} catch (ClassNotFoundException e) {
    e.printStackTrace();
} catch (NoSuchFieldException e) {
    e.printStackTrace();
} catch (IllegalAccessException e) {
    e.printStackTrace();
}
复制代码

参考文档ui

[1]stackoverflow How to handle :java.util.concurrent.TimeoutException: android.os.BinderProxy.finalize() timed out after 10 seconds errors?spa

[2]courtier博客 关于TimeoutException这件小事code

新增说明

形成这个错误的缘由应该是过于频繁的建立对象,最好检查项目中的代码确保没有这种状况。 一个被普遍使用的屏蔽系统字体大小设置的方法cdn

@Override
 public Resources getResources() {
        Resources res = super.getResources();
        Configuration config=new Configuration();
        config.setToDefaults();
        res.updateConfiguration(config,res.getDisplayMetrics());
        return res;
}
复制代码

这个方法将在每次getResources调用时new一个Configuration对象,这将致使部分机型来不及回收形成finalize() timed out错误。 正确的写法应该是写在生命周期方法中 如Activity的onCreate中对象

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    updateConfiguration();
    ......
}
private void updateConfiguration() {
    Resources res = getResources();
    Configuration config = new Configuration();
    config.setToDefaults();
    res.updateConfiguration(config, res.getDisplayMetrics());
}
复制代码
相关文章
相关标签/搜索