[GO]使用 CSTD(Code Self Test Development) 技术方式处理 error

背景知识

  • 在以前使用 VC 开发代码时,微软提供了 ASSERTVERIFY 宏,其在调试环境下能比较方便的发现问题。我基于此设计了 CSTD(Code Self Test Development) 和 API_VERIFY , COM_VERIFY 等宏帮助我开发了几乎 0bug 的 C/C++ 代码.
  • 在使用 go 语言开发时, 发现系统也是采用返回 error 的方式进行错误的处理, 而且不像 java, python 等使用异常。因此被戏称为 一半时间写代码,一半时间处理错误

Error 处理机制

  • 对于错误机制的处理上,编码人员一般有两种做法:
    • 对部分的返回值不予判断(直接 _ = xxx ),认为程序的运行不会出现那些错误。虽然代码清爽了,但实际运行环境下当程序中出现函数调用失败时,由于没有及时处理,就留下了Bug隐患,直到N久之后才发作。于是程序员就需要花费大量的时间、精力去再现、确认、更改Bug;
    • 对所有函数调用的地方都进行判断和处理。于是代码中出现大量的if…else等分支判断,造成程序的编写、维护工作量大幅上升,但是很多代码估计永远都不会执行(谁能告诉我正常情况下 File.Close() 什么时候会失败,失败后又该做什么?),而且往往在函数调用失败后,不判断具体的错误信息,只是简单的进行返回。没有日志的话,出现问题时很难定位。加日志的话,又到处都是日志。

CSTD(Code Self Test Development) 技术

  • 通过编写特定的 VerifyXxx 函数(Go等) /宏(C++),封装对指定函数的调用,自动检测函数的调用结果,在需要时打印日志、调用堆栈等,从而在发生问题时快速定位。
  • 函数定义如下:
func Verify(err error) error {if err != nil {checkAndHandleError(err, err.Error(), verifyAction, _SKIP_LEVEL)}return err
}func VerifyWithResult[T any](result T, err error) T {if err != nil {checkAndHandleError(err, err.Error(), verifyAction, _SKIP_LEVEL)}return result
}func VerifyWithResultEx[T any](result T, err error) (T, error) {if err != nil {checkAndHandleError(err, err.Error(), verifyAction, _SKIP_LEVEL)}return result, err
}func checkAndHandleError(err error, msg string, action CheckErrorAction, skip int) {if err != nil {fileName, lineNo, funName := flog.GetCallStackInfo(skip)msg := fmt.Sprintf("%s:%d (%s) FAIL(%s), msg=%s\n",fileName, lineNo, funName, reflect.TypeOf(err).String(), msg)switch action {case ACTION_LOG_ERROR:flog.Warnf(msg)case ACTION_FATAL_QUIT:panic(msg)}}
}
  • 其中 checkAndHandleError 是一个自定义的辅助函数, 可以在 error 不为 nil 时打印错误信息, 从而快速定位错误位置. 实际上的业务代码中即可使用如下的简单调用房室 :
// example: open a file should exist(local config file),
// if it not exists, then it's code error or CI/CD error, not runtime error.
func TestVerify(t *testing.T) {file := VerifyWithResult(os.Open("should_exist_conf_file"))defer func() {//Notice: when try to close a nil(*os.File), error with "invalid argument"_ = Verify(file.Close())}()//file.Read(xxxx)
}
  • 运行效果(可以看到代码中没有写日志的代码,但是程序发生问题时,能快速定位)
2024/01/29 21:31:54 [WARN] /path/to/go-library/debugutil/verify_test.go:13 (TestVerify) FAIL(*fs.PathError), msg=open should_exist_conf_file: The system cannot find the file specified.
2024/01/29 21:31:54 [WARN] /path/to/go-library/debugutil/verify_test.go:17 (func1) FAIL(*errors.errorString), msg=invalid argument

注意事项

  • VerifyXxx 只是帮助发现和更改错误的辅助机制,绝对不是错误处理逻辑。在发生错误时,一定要根据 error 进行后续的错误处理。对于大多数正常情况下不会、不该出错的代码,可以简单使用 VerifyXxx 即可,但对于可能出错的代码(如 os.Open(用户提供的路径) ),则需要进行错误处理.
  • 通常来说,合理使用 VerifyXxx 函数,能在很少投入的情况下(只需将原有代码中的“函数调用”换成“VerifyXxx(函数调用)”)即可发现和解决大部分的编码Bug,但如果结合敏捷开发中的TDD,将发挥更大威力。使用UT搭建自动化运行的框架并对功能进行测试,代码内部通过 VerifyXxx 进行测试。在分析、设计时仔细考虑一下,加上开发人员的责任心(实际上这才是实现0Bug程序的根本),再通过这两个工具的结合,实现出 0 Bug的程序将不再是梦想。

完整的源码位置:

  • https://gitee.com/fishjam/go-library/blob/main/debugutil/verify.go

补充信息

  • 目前采用代码中写死的 verifyAction 变量进行异常发生时的逻辑控制,感觉可以通过 build tags 的方式控制似乎更合适. 之后慢慢学习和调整吧.

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/655910.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

数据结构--树

一、树的基本术语 结点:树中的一个独立单元 结点的度:结点下分支的个数 树的度:树中所有结点中度的最大值 非终端结点:度不为0的结点 双亲和孩子:结点下的子树称为该结点的孩子.相应地,该结点称为孩子的双亲 兄弟:同一个双亲的孩子之间 祖先:从根到该结点所经分支上的所…

[NOIP2010 普及组] 接水问题

信息学奥赛一本通1950:【10NOIP普及组】接水问题 洛谷P1190 [NOIP2010 普及组] 接水问题 题解 【题目描述】 学校里有一个水房,水房里一共装有m个龙头可供同学们打开水,每个龙头每秒钟的供水量相等,均为1。 现在有n名同学准备接水…

猫头虎分享:Go语言开发资源库大全指南 - 从新手到专家

博主猫头虎的技术世界 🌟 欢迎来到猫头虎的博客 — 探索技术的无限可能! 专栏链接: 🔗 精选专栏: 《面试题大全》 — 面试准备的宝典!《IDEA开发秘籍》 — 提升你的IDEA技能!《100天精通鸿蒙》 — 从Web/安卓到鸿蒙大师!《100天精通Golang(基础入门篇)》 — 踏入Go…

【拦截器——详细介绍】

拦截器——详细介绍 1. 介绍2. 类型3. 工作流程4. 作用5. 实现方式6. 配置使用7. 注意事项 1. 介绍 拦截器(Interceptor)是一种在软件设计中常用的设计模式,通常用于在方法的执行前后进行额外的处理,比如日志记录、鉴权、缓存等。…

【Docker】docker安装jenkins

一、执行命令 下载jenkins镜像 #下载jenkins 镜像 docker pull jenkins/jenkins:latest-jdk8 启动jenkins容器 #启动jenkins 容器 #挂载 如果不挂载 每次启动jenkins的配置、插件、用户等信息都没有了 #jenkins_home 包含jenkins配置、插件、用户等信息。 要指定必须配置用…

【AI视野·今日NLP 自然语言处理论文速览 第七十六期】Fri, 12 Jan 2024

AI视野今日CS.NLP 自然语言处理论文速览 Fri, 12 Jan 2024 Totally 60 papers 👉上期速览✈更多精彩请移步主页 Daily Computation and Language Papers Axis Tour: Word Tour Determines the Order of Axes in ICA-transformed Embeddings Authors Hiroaki Yamagi…

linux离线升级openssh方法

检查openssh版本: 升级前openssh 版本为7.4 openssl 版本为1.0.2k Openssh9.6 所需openssl >1.1.1 因此openssl也需要升级。 为了防止升级失败,无法使用SSH登录,首先安装telnet 预防。查看是否安装了telnet 客户端及服务 未安装tel…

Java - JDBC

Java - JDBC 文章目录 Java - JDBC引言JDBC1 什么是JDBC2 MySQL数据库驱动3 JDBC开发步骤4 具体介绍 引言 思考: 当下我们如何操作数据库? 使用客户端工具访问数据库,手工建立连接,输入用户名和密码登录。编写SQL语句,点击执行…

Python学习之路-Django基础:模板

Python学习之路-Django基础:模板 模板 配置 在工程中创建模板目录templates。在settings.py配置文件中修改TEMPLATES配置项的DIRS值: TEMPLATES [{BACKEND: django.template.backends.django.DjangoTemplates,DIRS: [os.path.join(BASE_DIR, templates)], # 此…

Vue(十九):ElementUI 扩展实现树形结构表格组件的勾父选子、半勾选、过滤出半勾选节点功能

效果 原理分析 从后端获取数据后,判断当前节点是否勾选,从而判断是否勾选子节点勾选当前节点时,子节点均勾选全勾选与半勾选与不勾选的样式处理全勾选和全取消勾选的逻辑筛选出半勾选的节点定义变量 import {computed, nextTick, reactive, ref} from vue; import {tree} f…

解剖 Python 代码,深入学习 interpret 库的功能和应用!

更多资料获取 📚 个人网站:ipengtao.com Python是一门广泛应用的编程语言,拥有丰富的标准库和第三方库,可以用于各种应用场景。在Python中,有一个名为interpret的库,它提供了一种强大的方式来处理和执行Py…

LeetCode第605题 - 种花问题

题目 解答 方案一 public class Solution {public boolean canPlaceFlowers(int[] flowerbed, int n) {if (flowerbed null || flowerbed.length 0) {return false;}if (flowerbed.length < 2) {return (flowerbed[0] 1 ? 0 : 1) > n;}if (flowerbed.length < …

SpringBoot之@RequestParam注解

RequestParam &#xff08;org.springframework.web.bind.annotation.RequestParam&#xff09;用于将指定的请求参数赋值给方法中的形参。 有三个属性&#xff1a; &#xff08;1&#xff09;value&#xff1a;请求参数名&#xff08;必须配置&#xff09; &#xff08;2&…

【大数据】Flink 架构(三):事件时间处理

《Flink 架构》系列&#xff08;已完结&#xff09;&#xff0c;共包含以下 6 篇文章&#xff1a; Flink 架构&#xff08;一&#xff09;&#xff1a;系统架构Flink 架构&#xff08;二&#xff09;&#xff1a;数据传输Flink 架构&#xff08;三&#xff09;&#xff1a;事件…

<网络安全>《11 网络安全审计系统》

1 概念 审计是对资料作出证据搜集及分析&#xff0c;以评估企业状况&#xff0c;然后就资料及一般公认准则之间的相关程度作出结论及报告。 国际互联网络安全审计&#xff08;网络备案&#xff09;&#xff0c;是为了加强和规范互联网安全技术防范工作&#xff0c;保障互联网…

代码随想录训练营第三十一天|122.买卖股票的最佳时机II55.跳跃游戏45.跳跃游戏II

122.买卖股票的最佳时机II class Solution { public:int maxProfit(vector<int>& prices) {int earn0;for(int i 0; i < prices.size()-1;i){int x prices[i 1] - prices[i];if(x>0){earnx;}}return earn;} }; 55.跳跃游戏 本题关键在于看覆盖的范围 利…

基于C#制作一个俄罗斯方块小游戏

目录 引言游戏背景介绍游戏规则游戏设计与实现开发环境与工具游戏界面设计游戏逻辑实现游戏优化和测试性能优化测试工具和流程说明引言 俄罗斯方块是一款经典的益智游戏,深受玩家喜爱。本文将介绍如何使用C#编程语言制作一个简单的俄罗斯方块小游戏,并探讨其设计与实现过程。…

opencv-python 视频读取: VideoCapture.get()参数详解

视频读取demo import cv2 from tqdm import tqdmvideoCapture cv2.VideoCapture(video_path) if not videoCapture.isOpened(): # 若视频文件读取失败&#xff0c;读取下一段视频print(视频打开失败!!!)print(video_path)return False total_frames int(videoCapture.get(c…

【蓝桥杯日记】复盘篇二:分支结构

前言 本篇笔记主要进行复盘的内容是分支结构&#xff0c;通过学习分支结构从而更好巩固之前所学的内容。 目录 前言 目录 &#x1f34a;1.数的性质 分析&#xff1a; 知识点&#xff1a; &#x1f345;2.闰年判断 说明/提示 分析&#xff1a; 知识点&#xff1a; &am…

如何使用 Google 搜索引擎保姆级教程(附链接)

前言 需要挂梯子 一、介绍 "Google语法"通常是指在 Google 搜索引擎中使用一系列特定的搜索语法和操作符来精确地定义搜索查询。这些语法和操作符允许用户过滤和调整搜索结果&#xff0c;提高搜索的准确性。 二、安装 Google 下载 Google 浏览器 Google 官网ht…