Go 不是一种纯粹的面向对象编程语言。这段摘录自 Go 的常见问题解答,回答了 Go 是否是面向对象的问题。
是也不是。虽然Go有类型和方法,并且允许面向对象的编程风格,但是没有类型层次结构。Go中的“接口”概念提供了一种不同的方法,我们认为这种方法易于使用,并且在某些方面更通用。还有一些方法可以将类型嵌入到其他类型中,以提供类似于子类化(但不完全相同)的东西。此外,Go中的方法比c++或Java中的方法更通用:它们可以为任何类型的数据定义,甚至可以为内置类型(如普通的“未装箱”整数)定义。它们并不局限于结构体(类)。
此外,由于缺乏类型层次结构,Go中的“对象”感觉比c++或Java等语言要轻得多。
在接下来的教程中,我们将讨论如何使用 Go 实现面向对象的编程概念。与其他面向对象语言(例如 Java)相比,其中一些在实现上有很大不同。
Go 不提供类,但它提供结构体。可以在结构上添加方法。这提供了将数据和对数据进行操作的方法捆绑在一起的行为,类似于类。
让我们立即从一个例子开始,以便更好地理解。
我们将在此示例中创建一个自定义包,因为它有助于更好地理解结构如何有效替代类。
在里面创建一个子文件夹~/Documents/
并命名oop
。
让我们初始化一个名为 的 go 模块oop
。输入以下命令
go mod init oop
在里面创建一个子文件夹employee。在employee
文件夹内,创建一个名为employee.go
文件夹结构看起来像,
├── Documents
│ └── oop
│ ├── employee
│ │ └── employee.go
│ └── go.mod
请将employee.go
的内容替换为以下内容,
package employee
import (
"fmt"
)
type Employee struct {
FirstName string
LastName string
TotalLeaves int
LeavesTaken int
}
func (e Employee) LeavesRemaining() {
fmt.Printf("%s %s has %d leaves remaining\n", e.FirstName, e.LastName, (e.TotalLeaves - e.LeavesTaken))
}
在上面的程序中,第 1 行。指定该文件属于该employee
包。
在第7 行声明Employee 结构 ,在14行声明一个名为的方法LeavesRemaining
。现在我们有一个结构体和一个对结构体进行操作的方法,它们类似于类一样捆绑在一起。
在oop
文件夹内命名创建main.go
现在文件夹结构看起来像
├── Documents
│ └── oop
│ ├── employee
│ │ └── employee.go
│ ├── go.mod
│ └── main.go
main.go
内容如下
package main
import "oop/employee"
func main() {
e := employee.Employee {
FirstName: "Sam",
LastName: "Adolf",
TotalLeaves: 30,
LeavesTaken: 20,
}
e.LeavesRemaining()
}
我们在第 3 行导入employee
包。从第 12 行调用该结构体的方法LeavesRemaining()
该程序将打印输出,
Sam Adolf has 10 leaves remaining
我们上面编写的程序看起来不错,但其中有一个微妙的问题。让我们看看当我们用零值定义员工结构时会发生什么。将 main.go
的内容替换为以下代码,
package main
import "oop/employee"
func main() {
var e employee.Employee
e.LeavesRemaining()
}
我们所做的唯一更改是创建零值Employee
。该程序将输出,
has 0 leaves remaining
正如您所看到的,使用 0 值创建的变量Employee
是不可用的。它没有有效的名字、姓氏,也没有有效的休假详细信息。
在其他 OOP 语言(如 java)中,这个问题可以通过使用构造函数来解决。可以使用参数化构造函数创建有效的对象。
Go 不支持构造函数。如果类型的零值不可用,则程序员的工作是取消导出该类型以防止从其他包访问,并提供一个名为NewT(parameters)
的函数,该函数用所需的值初始化为类型T
。
Go 中的约定是命名一个创建NewT(parameters)
的函数。这将充当构造函数。如果包只定义了一种类型,那么 Go 中的约定是将此函数命名New(parameters)
让我们对我们编写的程序进行更改,以便每次创建员工时都可用。
第一步是取消导出Employee
结构并创建一个函数New()
来创建新的Employee
. 将代码替换employee.go
为以下内容,
package employee
import (
"fmt"
)
type employee struct {
firstName string
lastName string
totalLeaves int
leavesTaken int
}
func New(firstName string, lastName string, totalLeave int, leavesTaken int) employee {
e := employee {firstName, lastName, totalLeave, leavesTaken}
return e
}
func (e employee) LeavesRemaining() {
fmt.Printf("%s %s has %d leaves remaining\n", e.firstName, e.lastName, (e.totalLeaves - e.leavesTaken))
}
我们在这里做了一些重要的改变。我们在第 7行将 Employee 结构体的首字母改为小写。那就是我们已经改成type employee struct
。通过这样做,我们成功地取消了employee
结构的导出并阻止了其他包的访问。最好将未导出结构体的所有字段也设为未导出,除非有特定需要导出它们。由于我们不需要在包employee
外的任何地方访问结构体的字段employee
,因此我们也取消导出所有字段。
我们在LeavesRemaining()
方法中相应地更改了字段名称。
现在,由于employee
未导出,因此无法Employee
从其他包创建类型的值。因此,我们在第14 行提供了一个导出New
函数。它将所需的参数作为输入并返回新创建的员工。
该程序仍然需要进行更改才能使其正常工作,但让我们运行它来了解到目前为止更改的效果。如果运行该程序,它将失败并出现以下编译错误,
# oop
./main.go:6:8: undefined: employee.Employee
这是因为我们的employee
包中有一个未导出的employee
,文件并且无法从main
包中访问它。因此,编译器会抛出一个错误,指出该类型未在main.go
中定义。这正是我们想要的。现在没有其他包能够创建零值employee
。我们已成功阻止创建无法使用的员工结构值。现在创建员工的唯一方法是使用该New
函数。
将 main.go
的内容替换为以下内容,
package main
import "oop/employee"
func main() {
e := employee.New("Sam", "Adolf", 30, 20)
e.LeavesRemaining()
}
该文件的唯一更改是第 16 行。 我们通过将所需的参数传递给函数来创建一个新员工New
。
下面提供了进行所需更改后这两个文件的内容。
employee.go
package employee
import (
"fmt"
)
type employee struct {
firstName string
lastName string
totalLeaves int
leavesTaken int
}
func New(firstName string, lastName string, totalLeave int, leavesTaken int) employee {
e := employee {firstName, lastName, totalLeave, leavesTaken}
return e
}
func (e employee) LeavesRemaining() {
fmt.Printf("%s %s has %d leaves remaining\n", e.firstName, e.lastName, (e.totalLeaves - e.leavesTaken))
}
main.go
package main
import "oop/employee"
func main() {
e := employee.New("Sam", "Adolf", 30, 20)
e.LeavesRemaining()
}
运行该程序将输出,
Sam Adolf has 10 leaves remaining
因此你可以理解,虽然 Go 不支持类,但是可以有效地使用结构体来代替类,并且New(parameters)
可以使用签名方法来代替构造函数。
这就是 Go 中的类和构造函数。请留下您的宝贵意见和反馈。