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

本文深入分析了在OPPO系列机型上出现的finalize()timedout异常,探讨了其与定制ROM的关系,并提供了两种解决方案:关闭计时器或延长回收时间。同时,文章强调了频繁创建对象可能导致的问题,建议优化代码以避免此类错误。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在线上的APP经常收到如下的崩溃报告

.finalize() timed out

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

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

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();
}
复制代码

参考文档

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

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

新增说明

造成这个错误的原因应该是过于频繁的创建对象,最好检查项目中的代码确保没有这种情况。 一个被广泛使用的屏蔽系统字体大小设置的方法

@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());
}
复制代码

转载于:https://juejin.im/post/5c459d876fb9a049b780af81

<think>好的,我现在需要帮助用户理解Java方法`doAfterAllAnalysed`的功能和实现细节,以及涉及的类如ExcelAssetManager、AssetTypeEnum、LambdaQueryWrapper、BeanUtils和JwtUtil的作用。首先,我需要先理清楚用户提供的代码结构和相关引用信息。 用户提到了引用[1]、[2]、[3],这些引用主要涉及Apache Commons BeanUtils的功能,包括如何设置属性、类型转换以及使用条件。此外,用户还提到了其他类,如JwtUtil,可能用于处理JWT令牌。我需要将这些内容结合起来,分析`doAfterAllAnalysed`方法可能的逻辑。 首先,根据方法名称`doAfterAllAnalysed`,这通常是在数据处理完成后执行的操作,可能涉及数据持久化或后续处理。结合ExcelAssetManager,猜测这个方法可能是在Excel解析完成后,将数据存储到数据库或其他管理系统中。 接下来,AssetTypeEnum可能是一个枚举类,用于定义资产类型,帮助分类处理。LambdaQueryWrapper可能来自MyBatis-Plus,用于构建查询条件,进行数据库操作。BeanUtils用于属性复制,可能将解析后的Excel数据对象转换为实体对象。JwtUtil可能用于获取当前用户信息,以便记录操作者。 现在需要将这些元素整合起来,构建方法的具体流程。可能的步骤是: 1. 使用ExcelAssetManager获取解析后的资产数据列表。 2. 遍历每个资产对象,进行必要的数据处理。 3. 使用BeanUtils.copyProperties将Excel数据对象的属性复制到实体对象。 4. 通过JwtUtil获取当前用户,设置实体的创建人或相关字段。 5. 使用LambdaQueryWrapper构建查询条件,检查数据库中是否已存在相同资产,避免重复。 6. 根据检查结果,执行插入或更新操作。 同时,需要确保每个步骤符合引用中的BeanUtils使用条件,比如实体类必须有公共无参构造器和getter/setter方法。此外,异常处理和事务管理也是需要考虑的部分,但用户没有提到,可能需要进一步询问。 最后,用户可能希望了解各个类在方法中的作用,比如ExcelAssetManager负责管理Excel解析后的数据,AssetTypeEnum定义类型,LambdaQueryWrapper构建查询,BeanUtils处理属性复制,JwtUtil处理用户认证。需要逐一解释这些类的作用,并结合代码示例进行说明。 需要验证这些假设是否合理,是否有遗漏的细节。例如,是否有事务管理,如何处理异常情况,是否使用了其他框架如Spring。但根据现有信息,应该基于用户提供的引用和类名进行合理推断,并组织成清晰的解释。</think>### 分步解析`doAfterAllAnalysed`方法功能及实现 #### 一、方法核心功能 `doAfterAllAnalysed`方法通常在**Excel批量导入场景**中作为后置处理逻辑,主要完成: 1. 数据清洗与类型校验(通过`AssetTypeEnum`) 2. 数据转换(通过`BeanUtils`) 3. 数据持久化(通过`LambdaQueryWrapper`构建数据库操作) 4. 操作者身份记录(通过`JwtUtil`获取当前用户) #### 二、代码逻辑推测(需结合具体实现) 以下是典型的代码结构示例: ```java public void doAfterAllAnalysed(List<ExcelAssetDTO> excelData) { // 1. 获取当前操作用户 String username = JwtUtil.getCurrentUser(); // [^3] // 2. 遍历Excel解析结果 for (ExcelAssetDTO dto : excelData) { // 3. 类型校验 AssetTypeEnum type = AssetTypeEnum.parse(dto.getTypeName()); if (type == null) throw new IllegalArgumentException("非法资产类型"); // 4. DTO转Entity AssetEntity entity = new AssetEntity(); BeanUtils.copyProperties(dto, entity); // [^1][^2] entity.setCreator(username); // 记录操作者 // 5. 构建数据库查询条件 LambdaQueryWrapper<AssetEntity> wrapper = new LambdaQueryWrapper<>(); wrapper.eq(AssetEntity::getAssetCode, dto.getAssetCode()); // 6. 存在性检查与保存 if (excelAssetManager.exists(wrapper)) { excelAssetManager.update(entity, wrapper); } else { excelAssetManager.insert(entity); } } } ``` #### 三、关联类作用详解 1. **ExcelAssetManager** - **功能**:资产管理的服务层接口 - **典型方法**: ```java boolean exists(LambdaQueryWrapper<AssetEntity> wrapper); // 检查数据是否存在 void insert(AssetEntity entity); // 新增数据 void update(AssetEntity entity, LambdaQueryWrapper<AssetEntity> wrapper); // 更新数据 ``` 2. **AssetTypeEnum** - **功能**:资产类型枚举,保证数据有效性 - **典型实现**: ```java public enum AssetTypeEnum { SERVER(1, "服务器"), NETWORK(2, "网络设备"); public static AssetTypeEnum parse(String typeName) { return Arrays.stream(values()) .filter(e -> e.name.equals(typeName)) .findFirst().orElse(null); } } ``` 3. **LambdaQueryWrapper** - **作用**:MyBatis-Plus框架的条件构造器,通过Lambda表达式安全构建SQL条件 - **示例**: ```java wrapper.eq(AssetEntity::getStatus, 1) // WHERE status = 1 .likeRight(AssetEntity::getCode, "A-"); // AND code LIKE 'A-%' ``` 4. **BeanUtils** - **关键能力**:同名属性自动拷贝+类型转换 - 自动将`String`类型的`dto.id`转为`Long`类型的`entity.id` - 支持嵌套对象属性拷贝(需符合JavaBean规范)[^2] - **使用限制**: ```java // 要求ExcelAssetDTO和AssetEntity有匹配的字段名 public class ExcelAssetDTO { private String assetCode; // 自动映射到entity.assetCode // 必须提供getter/setter } ``` 5. **JwtUtil** - **典型实现**: ```java public class JwtUtil { // 从ThreadLocal或SecurityContext获取用户信息 public static String getCurrentUser() { return SecurityContextHolder.getContext().getAuthentication().getName(); } } ``` #### 四、性能优化建议 1. **批量操作优化**: ```java // 改用批量插入/更新 excelAssetManager.insertBatch(entityList); ``` 2. **异步处理**: ```java @Async // Spring异步注解 public void doAfterAllAnalysed(List<ExcelAssetDTO> excelData) {...} ``` #### 五、关键异常处理 | 异常类型 | 触发场景 | 处理建议 | |---------|---------|---------| | `NullPointerException` | BeanUtils拷贝空对象 | 增加非空校验 | | `ConversionException` | 字段类型转换失败 | 自定义Converter | | `DataIntegrityViolationException` | 数据库约束违反 | 前置数据校验 |
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值