Go 语言中创建方法
Go 不同于 java 或 C#,不支持类的概念。但是, Go 提供了一些面向对象编程(OOP)的特性和代码重用方法。其中一个特性是方法的概念,它可以实现 OOP 中的一些基本功能。
1. 使用方法
在 Go 语言中,方法是定义在特定数据类型上的函数。方法和普通函数之间唯一的区别在于方法有一个特殊的接收器类型,可以对其进行操作。为理解函数和方法之间的区别,可查看下面代码清单:
package mainimport "fmt"type account struct {Number stringBalance float64
}func (acct account) HaveEnoughBalance(value float64) bool {return acct.Balance >= value
}func HaveEnoughBalance2(acct account, value float64) bool {return acct.Balance >= value
}func main() {a := account{Number: "C95238436674394"}fmt.Println("Method result:", a.HaveEnoughBalance(150))fmt.Println("Function result:", HaveEnoughBalance2(a, 150))
}
在这段代码中,首先创建了一个名为 account 的结构体。然后,定义了一个函数 HaveEnoughBalance,它的输入为一个数值并返回一个布尔值。如果仔细观察,会发现在 func 关键字后面有一个(acct account) 语句。这告诉 Go,这个函数实际上是一个方法,且接收器是 account 类型。接着,又定义了一个传统的函数 HaveEnoughBalance2, 它的输入是一个 account 值,而不是一个接收器。
HaveEnoughBalance 和 HaveEnoughBalance2 有着相同的功能,但其实现方式不同。其中一个是一个带有 account 类型接收器的方法,另一个则是一个函数。它们的输出结果都是相同的,如下所示:
Method result: false
Function result: false
方法的函数之间的区别在于,方法包括一个接收器参数(acct account),它指示编译器通过 account 类型执行该方法,而函数需要 account 作为参数传递。使用接收器参数的方法的优点是 HaveEnoughBalance 方法现在可以通过 account 类型来执行。
2. 定义方法
可以在结构体上定义方法,但不限于此。实际上,可以在任何其他类型上定义方法。在 Go 中,可以使用以下语法基于内置类型创建新类型。
type identifier builtin_datatype
自定义类型的标识符必须遵循其他标识符使用的相同命名约定。在下面的代码清单中,使用关键字 type 基于现有的 string 类型创建了一个名为 s 的新类型。然后,将该类型用作一个方法的接收器。
package mainimport "fmt"type Text stringfunc (t Text) IsEmpty() bool {return len(t) <= 0
}func main() {text := Text("Hi")fmt.Println("type value:", text)fmt.Println("method value:", text.IsEmpty())
}
这个代码清单的执行结果如下所示:
type value: Hi
method value: false
为什么要创建一个类似字符串的新数据类型,而不是简单地将字符串类型传递给方法呢?通常,在Go 中,接收器类型的定义必须与方法在同一个包中。这意味着不能使用字符串作为接收器类型,因为它与正在创建的方法不在同一个包中。
要查看尝试使用字符串作为方法的接收器时会发生什么,可以用下面的代码