我是一名来自帝都985高校的机械研究生
本号专注于分享工科专业学习考试攻略、程序员学习求职经验、教师求职考试经验等相关内容,同时还会兼顾生活向,欢迎大家的关注和转发!
目录
错误处理
// Go 语言使用一个独立的、明确的返回值来传递错误信息的。这与
// 使用异常的 Java 和 Ruby 以及在 C 语言中经常见到的超重的
// 单返回值/错误值相比,Go 语言的处理方式能清楚的知道哪个函数
// 返回了错误,并能像调用那些没有出错的函数一样调用。
package main
import "errors"
import "fmt"
// 按照惯例,错误通常是最后一个返回值并且是 `error` 类
// 型,一个内建的接口。
func f1(arg int) (int, error) {
if arg == 42 {
// `errors.New` 构造一个使用给定的错误信息的基本
// `error` 值。
return -1, errors.New("can't work with 42")
}
// 返回错误值为 nil 代表没有错误。
return arg + 3, nil
}
// 通过实现 `Error` 方法来自定义 `error` 类型是可以的
// 这里使用自定义错误类型来表示上面的参数错误。
type argError struct {
arg int
prob string
}
func (e *argError) Error() string {
return fmt.Sprintf("%d - %s", e.arg, e.prob)
}
func f2(arg int) (int, error) {
if arg == 42 {
// 在这个例子中,我们使用 `&argError` 语法来建立一个
// 新的结构体,并提供了 `arg` 和 `prob` 这个两个字段的值。
return -1, &argError{arg, "can't work with it"}
}
return arg + 3, nil
}
func main() {
// 下面的两个循环测试了各个返回错误的函数。注意在 `if`
// 行内的错误检查代码,在 Go 中是一个普遍的用法。
for _, i := range []int{7, 42} {
if r, e := f1(i); e != nil {
fmt.Println("f1 failed:", e)
} else {
fmt.Println("f1 worked:", r)
}
}
for _, i := range []int{7, 42} {
if r, e := f2(i); e != nil {
fmt.Println("f2 failed:", e)
} else {
fmt.Println("f2 worked:", r)
}
}
// 你如果想在程序中使用一个自定义错误类型中的数据,你
// 需要通过类型断言来得到这个错误类型的实例。
_, e := f2(42)
if ae, ok := e.(*argError); ok {
fmt.Println(ae.arg)
fmt.Println(ae.prob)
}
}
协程
Go 语言中使用go关键字来创建并启动一个协程,「协程是一种轻量级的线程」,占用系统资源更少。
// _Go 协程_ 在执行上来说是轻量级的线程。
package main
import "fmt"
func f(from string) {
for i := 0; i < 3; i++ {
fmt.Println(from, ":", i)
}
}
func main() {
// 假设我们有一个函数叫做 `f(s)`。我们使用一般的方式
// 调并同时运行。
f("direct")
// 使用 `go f(s)` 在一个 Go 协程中调用这个函数。
// 这个新的 Go 协程将会并行的执行这个函数调用。
go f("goroutine")
// 你也可以为匿名函数启动一个 Go 协程。
go func(msg string) {
fmt.Println(msg)
}("going")
// 现在这两个 Go 协程在独立的 Go 协程中异步的运行,所以
// 我们需要等它们执行结束。这里的 `Scanln` 代码需要我们
// 在程序退出前按下任意键结束。
var input string
fmt.Scanln(&input)
fmt.Println("done")
}
我是一名来自帝都985高校的机械研究生
本号专注于分享工科专业学习考试攻略、程序员学习求职经验、教师求职考试经验等相关内容,同时还会兼顾生活向,欢迎大家的关注和转发!
文章评论