16.1 调试-日志、打印数据

1. 日志

日志是指程序执行过程中记录的信息。

  • 日志并非专为报告BUG而设,但可作为BUG发生时诊断故障的基础设施。
  • 日志通常采用文本文件的形式,便于直接阅读,以查找特定的事件或发生错误的原因

标准库的log包让应用程序能够将日志写入终端或文件

  • 日志中包含日期时间文本
    • log.Printf("This is a log message")
    • 2018/12/28 23:46:13 This is a log message
    • log.Printf()与前面我们学过的fmt.Printf()类似,也可以根据需要进行格式化的输出。
  • 致命错误发生时,调用Fatal函数,打印日志并终止程序运行。
    • log.Fatal(err)
// 打印日志
// 标准库的log包让应用程序能够将日志写入终端或文件。日志中包含日期、时间和文本
// 等
package mainimport "log"func main() {log.Printf("This is a log message")
}
// 打印输出:
// 2020/02/05 22:36:13 This is a log message

 

// 致命错误
// 标准库的log包还可用来记录程序运行过程中发生的致命错误。Fatal函数在打印日志
// 的同时终止程序的运行
package main
import ("errors""fmt""log"
)
func foo() error {return errors.New("We only just started and we are crashing")
}
func main() {err := foo()if err != nil {log.Fatal(err)// 至此,程序已“崩溃”后续程序不再执行}fmt.Println("We are going to bow out gracefully")
}
// 打印输出:
2018/12/28 23:52:38 We only just started and we are crashing
exit status 1
Process exiting with code: 1 

通过终端打印日志信息对于调试能起到一定作用,但想通过日志了解程序运行过程中发生的意外事件,则需要将其写入文件。

  • 通过Go语言本身提供的功能将日志写入文件
    • f, err := os.OpenFile("./file.log", os.O_CREATE|os.O_APPEND|os.O_RDWR, 0666)
    • log.SetOutput(f)
    • log.Printf("Log iteration %d", i) 
  • 借助操作系统提供的I/O重定向功能将日志写入文件
    • go run main.go 2>file.log
  • 最好通过操作系统将日志重定向到文件,而不要使用Go语言的文件操作,因为前者更加灵活,能够借助管道让其它工具在必要时参与对日志信息的分析。
// 写入日志文件
// 通过Go语言本身提供的功能将日志写入文件
package main
import ("log""os"
)
func main() {f, err := os.OpenFile("./file.log",os.O_CREATE|os.O_APPEND|os.O_RDWR, 0666)// 打开file.log文件,如不存在,则创建;如已存在,则追加新信息;可读可写if err != nil {log.Fatal(err)}defer f.Close()log.SetOutput(f)for i := 1; i <= 5; i++ {log.Printf("Log iteration %d", i)}
}
// 2020/02/06 09:21:44 Log iteration 1
// 2020/02/05 09:21:44 Log iteration 2
// 2020/02/05 09:21:44 Log iteration 3
// 2020/02/05 09:21:44 Log iteration 4
// 2020/02/05 09:21:44 Log iteration 5

 

// I/O重定向
// 借助操作系统提供的I/O重定向功能将日志写入文件 
// go run main.go 2>file.log
package main
import ("log"
)
func main() {for i := 1; i <= 5; i++ {log.Printf("Log iteration %d", i)}
}
// 打印输出:同“写入日志文件”

2. 打印数据

为了调试程序中的BUG,最快捷,最简单的方式就是将数据直接打印出来。

  • fmt.Printf("[DEBUG] text is: %q\n", text)

象这样打印数据的做法通常被视为一种快速解决方案,因为只需添加一行打印数据的代码,就可能使BUG立刻现身。

这样做的弊端是久而久之,程序中可能到处充斥着调试代码。

// 打印数据
// 为了调试程序中的BUG,最快捷,最简单的方式就是将数据直接打印出来
package main
import ("bufio""fmt""log""os""strings"
)
func main() {reader := bufio.NewReader(os.Stdin)	// 获取读取器fmt.Print("Guess the name of my " +"pet to win a prize: ")text, err := reader.ReadString('\n')// 获取标准输入的字符串文本if err != nil {log.Fatal(err)}// 打印调试信息 fmt.Printf("[DEBUG] text is: %q\n", text)if text == "john" {fmt.Println("You won! You win chocolate!")} else {fmt.Println("You didn't win. Better luck next time.")}
}
// 打印输出:
Guess the name of my pet to win a prize: John	// John是我们输入的
[DEBUG] text is: "John\r\n" // 输入字符串包含John,回车,换行,肯定与"john"// 不相等(大小写和多余字符),所以“猜错了”。
You didn't win. Better luck next time. 
// 打印数据
// 为了调试程序中的BUG,最快捷,最 
// 简单的方式就是将数据直接打印出来
package main
import ("bufio""fmt""log""os""strings"
)
func main() {reader := bufio.NewReader(os.Stdin)fmt.Print("Guess the name of my " +"pet to win a prize: ")text, err := reader.ReadString('\n')
if err != nil {log.Fatal(err)}text = strings.ToLower(strings.TrimSpace(text))	// 去掉结尾的空白字符;// 全小写text// 打印调试信息 fmt.Printf("[DEBUG] text is: %q\n", text)if text == "john" {fmt.Println("You won! You win chocolate!")} else {fmt.Println("You didn't win. Better luck next time.")}
}
// 打印输出:
Guess the name of my pet to win a prize: John
[DEBUG] text is: "john"
You won! You win chocolate! 

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

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

相关文章

python中操作文件的实践(2)

上一篇我总结了python对于文件操作的一些常用方法&#xff0c;这一篇主要记录对文件操作的一些常用方法 1.将文件中的内容进行替换 import os with open(python1.txt,encodingutf-8) as f1,\open(python1_bak.txt,encodingutf-8,modew) as f2:for line in f1:new_lineline.re…

Android驱动开发前的准备

查看Linux内核版本 Android系统中的“ 设备 ” > “关于手机 ” 中查看当前Android系统所采用的Linux内核版本 使用uname命令查看Linux内核版本 # Ubuntu下 , Linux内核的版本 uname -a查看proc/version文件获取Linux内核版本 cat /proc/version/proc不是普通的文件系统…

目标检测——叶片计数数据集

引言 亲爱的读者们&#xff0c;您是否在寻找某个特定的数据集&#xff0c;用于研究或项目实践&#xff1f;欢迎您在评论区留言&#xff0c;或者通过公众号私信告诉我&#xff0c;您想要的数据集的类型主题。小编会竭尽全力为您寻找&#xff0c;并在找到后第一时间与您分享。 …

R语言绘图 --- 桑基图(Biorplot 开发日志 --- 5)

「写在前面」 在科研数据分析中我们会重复地绘制一些图形&#xff0c;如果代码管理不当经常就会忘记之前绘图的代码。于是我计划开发一个 R 包&#xff08;Biorplot&#xff09;&#xff0c;用来管理自己 R 语言绘图的代码。本系列文章用于记录 Biorplot 包开发日志。 相关链接…

VUE3 学习笔记(14):VUE3 组合式API与传统选项式API用法

VUE3相较VUE2的亮点很多&#xff0c;作为后端开发置于前端最大的感受就是组合式API&#xff08;之前采用的是选项式API&#xff09;&#xff1b;它使得整体更简洁易用,但值得提醒的是官方并未强制要求二选一&#xff0c;尽管如此在同一个项目中还是不要出现两种写法。 选项式AP…

详解大模型微调数据集构建方法(持续更新)

大家好,我是herosunly。985院校硕士毕业,现担任算法t研究员一职,热衷于机器学习算法研究与应用。曾获得阿里云天池比赛第一名,CCF比赛第二名,科大讯飞比赛第三名。拥有多项发明专利。对机器学习和深度学习拥有自己独到的见解。曾经辅导过若干个非计算机专业的学生进入到算…

【通知】上市公司嵌入式工程师带队授课,成品展示~~

1&#xff0c;成品展示&#xff1a; 2&#xff0c;产品需求&#xff1a; 设计一款无线CAN转发器&#xff0c;由若干个终端组成&#xff0c;若干个终端之间可以将接收到的CAN数据通过无线的方式转发出去&#xff0c;在复杂的条件下&#xff0c;传输距离不低于200m。 该CAN转发器…

vscode+latex设置跳转快捷键

安装参考 https://blog.csdn.net/Hacker_MAI/article/details/130334821 设置默认recipe ctrl P 打开设置&#xff0c;搜索recipe 也可以点这里看看有哪些配置 2 设置跳转快捷键

leetcode7 整数翻转

给你一个 32 位的有符号整数 x &#xff0c;返回将 x 中的数字部分反转后的结果。 如果反转后整数超过 32 位的有符号整数的范围 [−231, 231 − 1] &#xff0c;就返回 0。 假设环境不允许存储 64 位整数&#xff08;有符号或无符号&#xff09;。 示例 1&#xff1a; 输入…

[AIGC] @Component与@Bean的区别详解

在Spring框架中&#xff0c;Component和Bean常常被用来进行依赖注入&#xff0c;让Spring自身管理对象的生命周期。然而&#xff0c;尽管它们将目标类标记为Spring容器中的构件&#xff0c;但是二者的应用场景和工作方式存在重要的区别。 文章目录 ComponentBean主要区别 Compo…

IIC信号质量测试、时序测试详解

IIC 时序图 信号质量测试 1、vIL: 低输入电平。 2、vIH: 高输入电平。 3、vhys: 施密特触发器输入的滞后。 4、vOL1: VDD>2V时&#xff0c;低电平输出电压&#xff08;漏极开路或集电极开路&#xff09;。 5、vOL3: VDD<2V时&#xff0c;低电平输出电压&#xff08;漏极开…

【Vue】水果购物车-基本渲染

在Vue中&#xff0c;如果想要实现全选反选功能&#xff0c;通常使用计算属性 完整代码 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta http-equiv"X-UA-Compatible" content"IEedge&qu…

Spring中的事务是如何实现的

Spring里的事务管理&#xff0c;你可以想象成是银行里的一个业务办理过程。就像你在银行办转账时&#xff0c;要么全部完成&#xff08;钱从一个账户转到另一个账户&#xff09;&#xff0c;要么什么都不变&#xff08;如果中间出问题了&#xff0c;钱不会丢失&#xff09;。Sp…

JMeter Plugins Manager---插件安装

参考文章&#xff1a;https://blog.51cto.com/u_14126/6291032 需求&#xff1a; 安装【jpgc - Standard Set】插件 常用插件&#xff1a; 点击下载–报错如下&#xff1a; Failed to apply changes:Cannotapplychanges:Haveno write accessforJMeterdirectories,notpossib…

Element Plus使用总结

一、引言 Element Plus是一套为开发者、设计师和产品经理准备的基于Vue 3的组件库&#xff0c;它继承了Element UI的优点&#xff0c;同时进行了诸多改进和优化&#xff0c;使其更加适用于现代Web应用的开发。以下是对Element Plus使用的总结。 二、安装与引入 安装&#xf…

JavaScript 中创建函数的多种方式

在 JavaScript 中&#xff0c;可以通过多种方式创建函数。每种方式都有其特定的用途、优点和缺点&#xff0c;以及适用的使用场景。以下是几种常见的创建函数的方式及其详细说明。 1. 函数声明&#xff08;Function Declaration&#xff09; 示例 function add(a, b) {retur…

Python版《消消乐》,附源码

曾经风靡一时的消消乐&#xff0c;至今坐在地铁上都可以看到很多人依然在玩&#xff0c;想当年我也是大军中的一员&#xff0c;那家伙&#xff0c;吃饭都在玩&#xff0c;进入到高级的那种胜利感还是很爽的&#xff0c;连续消&#xff0c;无限消&#xff0c;哈哈&#xff0c;现…

60、最大公约数

最大公约数 题目描述 给定n对正整数ai,bi&#xff0c;请你求出每对数的最大公约数。 输入格式 第一行包含整数n。 接下来n行&#xff0c;每行包含一个整数对ai,bi。 输出格式 输出共n行&#xff0c;每行输出一个整数对的最大公约数。 数据范围 1 ≤ n ≤ 1 0 5 , 1≤n≤…

0基础学习区块链技术——去中心化

“去中心化”是区块链技术的核心。那么我们该如何理解这个概念呢&#xff1f; 我们可以假想在一次现实转账中&#xff0c;有哪些“中心化”的行为&#xff1a; 判断余额是否足够。即判断转出的钱是否少于账户里剩余的钱&#xff0c;能够判断的是账户所在的银行。 如果余额足够…

读AI未来进行式笔记03自然语言处理技术

1. AI伙伴 1.1. 作为AI能力的集大成者&#xff0c;AI伙伴融合了各种复杂的AI技术 1.2. 人类唯一可能超越AI的领域&#xff0c;只可能在机器无法触及之处&#xff0c;那是属于人类感性与直觉的领域 1.3. 要读懂人类&#xff0c;需要漫长而平缓的学习过程 1.4. AI塑造了我们&…