异常之道:探索Java异常处理与日志的黄金准则

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: 异常之道:探索Java异常处理与日志的黄金准则

异常处理与日志规范

  1. 底层原始异常不要捕获直接向外抛出,一直往外抛出,直到 Service 或者 Controller;
  2. 自定义的 BusinessException 异常统一在 Controller 层进行汇总处理;
  3. 其他异常需要首先在 Service 层进行 error 级别日志记录,之后向外层抛出 BusinessException 并调用 setCause 设置造成原因(注意 Service 只能向外抛出 BusinessException,其他异常需要进行异常转换);
  4. Controller 层不要再向外抛出异常,需要在 Controller 层进行捕获操作
  • 如果返回的是 RespResult,在 catch 中返回 RespResult.fail
  • 如果返回的是页面,在 catch 中返回 "redirect:/500"
  • Controller 层捕获异常的时候不需要打印错误日志
  1. 系统只处理受检异常,运行时异常不要捕获也不要抛出,属于Bug,应当解决;
  2. 进行异常转换时都要进行 error 级别日志记录,并且调用 initCause 方法设置错误原因;
  3. Service 层向外抛出自定义的 BusinessException 异常,由调用此 Service 的 Controller 或者定时任务捕获处理;
  4. 抛出定义的 CdnHuaweiException 或 BusinessException 是可以使用链式调用的方式快捷打印日志,示例如下:
  • throw new CdnHuaweiException("XXXX发生错误!错误原因:{}", e.getMessage()).setCause(e).log();
  1. 其他特殊类(Bean、Component、Interceptor…)的异常处理和日志打印规则可自行处理;

程序示例

DomainConfigureApi.java

public static JSONObject getDomainConfigs(String domainName) throws CdnHuaweiException {
    try {
        Request request = HuaweiRequest.getRequest(CdnConst.GET_DOMAIN_CONFIGS.replace("{domain_name}", domainName), "GET");
        Response response = HuaweiRequest.doRequest(request);
        return HuaweiRequest.dealResponse(response);
    } catch (Exception e) {
        log.error("查询域名配置接口失败!错误原因:{}", e.getMessage());
        throw new CdnHuaweiException("查询域名配置接口失败!错误原因:{}", e.getMessage()).setCause(e);
        // 如果打印的日志和报错的信息一样,那么上面两行代码也可以使用下面代替
        // throw new CdnHuaweiException("查询域名配置接口失败!错误原因:{}", e.getMessage()).setCause(e).log();
    }
}

CdnDomainService.java

public JSONObject getDomainConfig(String domainName) throws BusinessException {
    JSONObject jsonObject = null;
    try {
        jsonObject = DomainConfigureApi.getDomainConfigs(domainName);
    } catch (CdnHuaweiException e) {
        log.error("获取域名的详细配置失败,域名:{}", domainName);
        throw new CdnHuaweiException("获取域名的详细配置失败!错误原因:{}", e.getMessage()).setCause(e);
    }
    JSONObject configs = jsonObject.getJSONObject("configs");
    // 解析主源站
    JSONArray sourcesArray = configs.getJSONArray("sources");
    for (int i = 0; i < sourcesArray.size(); i++) {
        JSONObject source = sourcesArray.getJSONObject(i);
        if (source.getInteger("priority") == SourcePriority.MAIN) {
            configs.put("main_source", source);
        }
        if (source.getInteger("priority") == SourcePriority.BACK) {
            configs.put("back_source", source);
        }
    }
    return configs;
}

DomainSettingsPageController.java

@GetMapping("/domain-setting-basic")
public String domainBasicSettings(Long id, Map<String, Object> map) {
    if (Assert.isEmpty(id)) {
        return "redirect:/400";
    }
    CdnDomainVo cdnDomainVo = cdnDomainService.getCdnDomainVoById(id);
    if (Assert.isEmpty(cdnDomainVo)) {
        return "redirect:/404";
    }
    // 获取域名的详细配置
    try {
        JSONObject domainConfig = cdnDomainService.getDomainConfig(cdnDomainVo.getDomainName());
        map.put("domainConfig", domainConfig);
    } catch (BusinessException e) {
        return "redirect:/500";
    }
    // 查询所有的业务类型
    List<CdnBusinessType> businessTypes = cdnBusinessTypeService.queryAll();
    // 查询所有的业务范围
    List<CdnServiceArea> serviceAreas = cdnServiceAreaService.queryAll();
    // 查询所有源站类型
    List<CdnOriginType> originTypes = cdnOriginTypeService.queryAll();
    map.put("businessTypes", businessTypes);
    map.put("serviceAreas", serviceAreas);
    map.put("originTypes", originTypes);
    map.put("domain", cdnDomainVo);
    return "admin/domain/domain-setting-basic";
}


相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
相关文章
|
18天前
|
Java
java 多线程异常处理
本文介绍了Java中ThreadGroup的异常处理机制,重点讲解UncaughtExceptionHandler的使用。通过示例代码展示了当线程的run()方法抛出未捕获异常时,JVM如何依次查找并调用线程的异常处理器、线程组的uncaughtException方法或默认异常处理器。文章还提供了具体代码和输出结果,帮助理解不同处理器的优先级与执行逻辑。
|
2月前
|
SQL Java 中间件
【YashanDB知识库】yasdb jdbc驱动集成BeetISQL中间件,业务(java)报autoAssignKey failure异常
在BeetISQL 2.13.8版本中,客户使用batch insert向yashandb表插入数据并尝试获取自动生成的sequence id时,出现类型转换异常。原因是beetlsql在prepareStatement时未指定返回列,导致yashan JDBC驱动返回rowid(字符串),与Java Bean中的数字类型tid不匹配。此问题影响业务流程,使无法正确获取sequence id。解决方法包括:1) 在batchInsert时不返回自动生成的sequence id;2) 升级至BeetISQL 3,其已修正该问题。
【YashanDB知识库】yasdb jdbc驱动集成BeetISQL中间件,业务(java)报autoAssignKey failure异常
|
2月前
|
SQL druid Oracle
【YashanDB知识库】yasdb jdbc驱动集成druid连接池,业务(java)日志中有token IDENTIFIER start异常
客户Java日志中出现异常,影响Druid的merge SQL功能(将SQL字面量替换为绑定变量以统计性能),但不影响正常业务流程。原因是Druid在merge SQL时传入null作为dbType,导致无法解析递归查询中的`start`关键字。
|
2月前
|
运维 Java 程序员
Java中的异常处理方法
本文深入剖析Java异常处理机制,介绍可检查异常、运行时异常和错误的区别与处理方式。通过最佳实践方法,如使用合适的异常类型、声明精确异常、try-with-resources语句块、记录异常信息等,帮助开发者提高代码的可靠性、可读性和可维护性。良好的异常处理能保证程序稳定运行,避免资源泄漏和潜在问题。
108 5
|
3月前
|
Java 程序员 开发者
Java社招面试题:一个线程运行时发生异常会怎样?
大家好,我是小米。今天分享一个经典的 Java 面试题:线程运行时发生异常,程序会怎样处理?此问题考察 Java 线程和异常处理机制的理解。线程发生异常,默认会导致线程终止,但可以通过 try-catch 捕获并处理,避免影响其他线程。未捕获的异常可通过 Thread.UncaughtExceptionHandler 处理。线程池中的异常会被自动处理,不影响任务执行。希望这篇文章能帮助你深入理解 Java 线程异常处理机制,为面试做好准备。如果你觉得有帮助,欢迎收藏、转发!
218 14
|
3月前
|
缓存 Java 应用服务中间件
java语言后台管理若依框架-登录提示404-接口异常-系统接口404异常如何处理-登录验证码不显示prod-api/captchaImage 404 (Not Found) 如何处理-解决方案优雅草卓伊凡
java语言后台管理若依框架-登录提示404-接口异常-系统接口404异常如何处理-登录验证码不显示prod-api/captchaImage 404 (Not Found) 如何处理-解决方案优雅草卓伊凡
586 5
|
3月前
|
XML JSON Java
Java中Log级别和解析
日志级别定义了日志信息的重要程度,从低到高依次为:TRACE(详细调试)、DEBUG(开发调试)、INFO(一般信息)、WARN(潜在问题)、ERROR(错误信息)和FATAL(严重错误)。开发人员可根据需要设置不同的日志级别,以控制日志输出量,避免影响性能或干扰问题排查。日志框架如Log4j 2由Logger、Appender和Layout组成,通过配置文件指定日志级别、输出目标和格式。
|
4月前
|
Java 数据库连接 数据处理
探究Java异常处理【保姆级教程】
Java 异常处理是确保程序稳健运行的关键机制。它通过捕获和处理运行时错误,避免程序崩溃。Java 的异常体系以 `Throwable` 为基础,分为 `Error` 和 `Exception`。前者表示严重错误,后者可细分为受检和非受检异常。常见的异常处理方式包括 `try-catch-finally`、`throws` 和 `throw` 关键字。此外,还可以自定义异常类以满足特定需求。最佳实践包括捕获具体异常、合理使用 `finally` 块和谨慎抛出异常。掌握这些技巧能显著提升程序的健壮性和可靠性。
79 4
Java基础异常-自定义异常
Java基础异常-自定义异常
Java基础异常-自定义异常
|
Java 编译器
Java中的异常(抛出异常、自定义异常等)
Java中的异常(抛出异常、自定义异常等)
435 1