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

简介: 异常之道:探索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";
}


相关实践学习
【涂鸦即艺术】基于云应用开发平台CAP部署AI实时生图绘板
【涂鸦即艺术】基于云应用开发平台CAP部署AI实时生图绘板
相关文章
|
6月前
|
安全 Java
Java异常处理:程序世界的“交通规则
Java异常处理:程序世界的“交通规则
371 98
|
6月前
|
安全 Java 编译器
驾驭Java异常处理:从新手到专家的优雅之道
驾驭Java异常处理:从新手到专家的优雅之道
300 59
|
9月前
|
Java 编译器 数据库连接
Java异常处理:写出更健壮的代码
Java异常处理:写出更健壮的代码
255 0
|
8月前
|
Java 数据库 C++
Java异常处理机制:try-catch、throws与自定义异常
本文深入解析Java异常处理机制,涵盖异常分类、try-catch-finally使用、throw与throws区别、自定义异常及最佳实践,助你写出更健壮、清晰的代码,提升Java编程能力。
|
10月前
|
运维 监控 安全
Syslog 日志分析与异常检测技巧
系统日志蕴含设备运行关键信息,但分析提取颇具挑战。本文详解从命令行工具(如 Grep、Tail、Awk)到专业软件(如 EventLog Analyzer)的全流程日志分析技巧,助你高效挖掘 Syslog 价值,提升运维与安全响应能力。
577 4
|
10月前
|
Java 程序员 数据库连接
我们详细地讲解一下 Java 异常及要如何处理
我是小假 期待与你的下一次相遇 ~
222 1
|
11月前
|
Java
java 多线程异常处理
本文介绍了Java中ThreadGroup的异常处理机制,重点讲解UncaughtExceptionHandler的使用。通过示例代码展示了当线程的run()方法抛出未捕获异常时,JVM如何依次查找并调用线程的异常处理器、线程组的uncaughtException方法或默认异常处理器。文章还提供了具体代码和输出结果,帮助理解不同处理器的优先级与执行逻辑。
232 1
|
SQL druid Oracle
【YashanDB知识库】yasdb jdbc驱动集成druid连接池,业务(java)日志中有token IDENTIFIER start异常
客户Java日志中出现异常,影响Druid的merge SQL功能(将SQL字面量替换为绑定变量以统计性能),但不影响正常业务流程。原因是Druid在merge SQL时传入null作为dbType,导致无法解析递归查询中的`start`关键字。
|
监控 安全 Apache
什么是Apache日志?为什么Apache日志分析很重要?
Apache是全球广泛使用的Web服务器软件,支持超过30%的活跃网站。它通过接收和处理HTTP请求,与后端服务器通信,返回响应并记录日志,确保网页请求的快速准确处理。Apache日志分为访问日志和错误日志,对提升用户体验、保障安全及优化性能至关重要。EventLog Analyzer等工具可有效管理和分析这些日志,增强Web服务的安全性和可靠性。
521 9