这是一个为 Go 添加 !err 错误处理语法糖的项目, 与 Go 完全兼容(毕竟只是语法糖)
有了它你可以写如下代码, 代码更加紧凑 阅读更轻松
转换前:
package main
import (
  "fmt"
  "os"
)
func main() {
  body, !err := readSelf()
  fmt.Println("main.go content")
  fmt.Println(body)
}
func readSelf() (content string, err error) {
  body, !err := os.ReadFile("main.go")
  content = string(body)
  return
}
转换后:
package main
import (
  "fmt"
  "os"
)
func main() {
  body, err := readSelf()
  if err != nil {
    return
  }
  fmt.Println("main.go content")
  fmt.Println(body)
}
func readSelf() (content string, err error) {
  body, err := os.ReadFile("main.go")
  if err != nil {
    return
  }
  content = string(body)
  return
}
项目地址: https://github.com/gone-lang/gone
先来问问各位大佬的意见, 看下怎么做比较好
打印错误日志.
func readSelf() (content string, err error) {
	defer func() {
		if err == nil {
			return
		}
		slog.Error("wrong", "err", err)
	}()
	body, ierr := os.ReadFile("main.go")
	content = string(body)
	return
}
如果需要立即处理错误, 可以不使用语法糖使用原有的 if err != nil {} 模式.
语法糖是可以随时退出不使用的
|  |      1AKAUP      2023-08-31 08:19:24 +08:00  1 照你的方法,如果在报错的想打印日志应该怎么做呢,比如: ``` if err != nil { logger.Error("something wrong") return err } ``` | 
|  |      2morri      2023-08-31 08:24:56 +08:00 个人感觉其实 没必要每个错误都去处理,关键点判断好,单元测试写好,应该就可以了。 例如 var u entity.User _=dao.User.Ctx(ctx).Scan(&u,"id",uid) if u.id==0{ return fmt.error("user does not exist") } // 这里直接返回自定义的错误,中间件也好其他处理,比如多语言返回 | 
|  |      3rekulas      2023-08-31 08:29:50 +08:00 if err != nil { return } 好家伙 返回值都不要了? | 
|  |      5rekulas      2023-08-31 08:37:29 +08:00  1 有个大问题是带来了兼容性问题, 我喜欢 go 的就是一次编码到处运行,如果用这种可能就得变成一次编码到处配置环境..用户下载一份源码,发现还无法编译得单独配置下,妥妥有前些年好压的既视感了,除非官方支持 | 
|      6sofukwird OP @AKAUP 使用 defer 打印错误, 如果需要立即处理错误, 可以不是用语法糖使用原有的 `if err != nil {}` 模式. 语法糖是可以随时退出不使用的 ```go func readSelf2() (content string, err error) { defer func() { slog.Error("wrong", "err", err) }() body, !err := os.ReadFile("main.go") content = string(body) return } ``` @rekulas 并不会出现兼容性问题, 会另外生成一个转译后 go 文件, 与其他 go 库是无缝操作的, 你可以认为是 typescript 这种 @morri 其实你这里也是进行了一个错误处理的判断 | 
|  |      7lisxour      2023-08-31 09:00:54 +08:00  1 if err != nil { 不带其他逻辑直接 return } 一个语法糖只为了对这么一个特定场景下少两三行代码,大概率不会给你过的 | 
|  |      8proxychains      2023-08-31 09:02:05 +08:00 @morri 头像好评 | 
|      9cin      2023-08-31 09:08:09 +08:00  4 | 
|  |      11SingeeKing PRO 我很好奇附言里是怎么把 !err 打成 ierr 的…… 难道 lz 用的手写输入? | 
|      13sofukwird OP @SingeeKing 这个是一个已有的转译实现, 我用了 `ierr` 作为 `!err` 的替代, 但遇到了一些问题, 打断点需要到转译后的文件才能打上, 在开发中还蛮影响思路的, 后面想到了如果做成 golang 超集语言的话就可以打上断点, 于是又起了个仓库准备实现新思路 已有的转译实现: https://github.com/shynome/err4 , 现在正在用 (就像前段时间的帖子说的, 每个程序员都有改造语言的冲动, 我也是对 golang error 的错误处理不满意所以就有这么一个个尝试 | 
|      14lanlanye      2023-08-31 09:28:19 +08:00 via iPhone  4 真要做的话我觉得 rust 的方式就不错,语句末尾加一个问号。 | 
|      15githmb      2023-08-31 10:03:30 +08:00 Rust: let Ok(bar) = foo() else { return fuckyou } | 
|  |      16ganbuliao      2023-08-31 10:23:46 +08:00 按照 go 的 习惯 是不会加新的语法糖的  毕竟 go 写出来的代码 主要是为了 review 的人 看着舒服的 | 
|      17vchroc      2023-08-31 10:25:25 +08:00 语言肯定朝着越来越好用的方向发展。 错误处理,目前看比较好用的有两种: GOTO:Exception/Panic 正确错误二元组:Result<T, E>( Rust ) golang err 绝对不是正确的方向 | 
|      18neoblackcap      2023-08-31 10:27:03 +08:00 @lanlanye 有类型系统兜底呢,不是简单语法改改就能模仿 | 
|  |      19fioncat      2023-08-31 10:31:05 +08:00 像 Rust 那样整个?作为语法糖? err := handle() if err != nil { return fmt.Errorf("Handle error: %w", err) } ===> handle()?.Errorf("Handle error") | 
|  |      20mcfog      2023-08-31 10:51:30 +08:00 https://github.com/golang/go/issues/32437 这里有最全的 golang error handle 的意见 | 
|  |      21learningman      2023-08-31 10:53:24 +08:00 via Android 你这样不如 v, ? := xxx() | 
|  |      22guonaihong      2023-08-31 11:16:44 +08:00 现在一堆 ai 辅助编程的插件 感觉收益不是那么明显。copilot 一个回车解决的事情。 | 
|      23codersdp1      2023-08-31 11:46:19 +08:00 func readSelf() (string, error) { body, err := os.ReadFile("main.go") if err != nil { return "",err } content = string(body) return content,nil } 这种怎么处理 🤔️ | 
|  |      24PTLin      2023-08-31 12:32:03 +08:00 https://github.com/golang/go/issues?q=label%3Aerror-handling 看看这些数不清的提案,相信总有一个是比你这个想法优秀且被枪毙了的。 | 
|      25sofukwird OP @codersdp1 #23 这种情况不使用语法糖,直接使用你现在使用的代码,这个糖是有限的,不打算应对所有情况 @guonaihong 主要还是想阅读的时候更简单轻松,虽然编辑器能补全但代码行数是实打实地变多了 @fioncat 不选择 v := xxx()? 是因为它假定了最后返回 error ,而且在只返回 error 的情况下可读性并不好,需要看到末尾才知道这行是否可能出错 xxx()? @learningman 不选择 v, ? := xxx() 则是因为转译后需要自己生成 err 临时变量名并返回,临时变量名会千奇百怪的导致可读性下降。那使用 v, ?err := xxx() ?我一开始就是提议的这种后来发现 v, !err := xxx() 这种可读性更好。 我选择这种模式是因为它易于实现,又或者说正是因为它易于实现我才能选择它 | 
|  |      26raies      2023-08-31 13:35:56 +08:00 不如这样 body := readSelf()? | 
|  |      27mogita      2023-08-31 13:48:51 +08:00  1 感觉无糖更方便,因为我已经告诉别人我是 Errlang 开发者了。 | 
|  |      28index90      2023-08-31 16:16:28 +08:00 FP 才是最优解: type Result[T any] struct { v T; err error } func(r *Result[T]) IfErr(f func(error)) { if r.err != nil {f(err)} } func fmap[A, B any](ra Result[A], f f(a) Result[B]) Result[B] { if ra.err != nil { return f(ra.v) } return Result[B]{err: error} } | 
|      29liuguang      2023-08-31 16:23:48 +08:00 设计上用元组就是一个大的失误,值和错误不会同时出现。 而 rust 的枚举就是很好的设计,rust 的问号运算符也是一个好的语法糖。 | 
|  |      30mcfog      2023-08-31 16:59:48 +08:00 | 
|  |      31Mohanson      2023-08-31 18:25:01 +08:00  1 对于编程语言来说, 除非你有充分的理由, 否则不要加语法糖.  注: 我觉得这样写很 cool 就不是一个充分的理由. | 
|  |      32mainjzb      2023-08-31 18:46:01 +08:00 | 
|  |      33aduo      2023-08-31 19:24:53 +08:00 想法挺好的... 但是没有明确的 return 语句却可能会 return ,有点孩怕, 感觉容易出事故 | 
|      34sofukwird OP | 
|      35joesonw      2023-08-31 22:24:03 +08:00 via iPhone 最关键的问题是,一般是鼓励 wrap 一下错误,不然你只看到 io 错误,根本不知道是哪里的问题,排查起来头痛。 | 
|  |      36mainjzb      2023-09-01 09:41:32 +08:00  1 我也会拒绝你,你这等于给语言加了一个宏仅仅为了偷懒少写一些代码而已。 当然 go 的初期设计确实考虑的很简陋,更聪明一点的做法是像 zig 或 rust 那样。 go error 设计更大的问题是没有保存堆栈。对于仅仅多写一点代码的问题,完全可以用 ide 来自动补充来解决。 无论是你的设计还是 rust 的设计还是 zig 的设计,还是当前 go 的设计,都不影响阅读代码。 写代码只是为了偷懒,我认为没有完全的必要性添加此项更改。 |