> 关注公众号【爱发白日梦的后端】分享技术干货、读书笔记、开源项目、实战经验、高效开发工具等,您的关注将是我的更新动力!
在Go语言中,虽然没有像面向对象语言那样的类,但通过结构体类型和方法,仍然支持部分面向对象编程(OOP)的概念。
## 封装(Encapsulation)
封装是一种将一个对象的实现细节隐藏起来,使其对其他对象不可见的做法,这样可以实现解耦。
例如,考虑以下结构体:
```go
type Student struct{
name string
rollNo uint64
}
```
其中的`name`和`rollNo`字段是私有的,因为它们以小写字母开头。为了提供公共访问,可以为这些字段定义对应的`getter`和`setter`方法。
```go
func (s *Student) GetName() string {
return s.name
}
func (s *Student) SetName(name string) {
s.name = name
}
func (s *Student) GetRollNo() uint64 {
return s.rollNo
}
func (s *Student) SetRollNo(roll uint64) {
s.rollNo = roll
}
```
现在,程序的其他部分也可以创建`Student`结构体的对象,并通过公共的`getter`和`setter`方法访问`name`和`rollNo`,从而实现了封装。
## 抽象(Abstraction)
数据抽象是一种设计模式,其中数据仅对语义相关的函数可见,以防止误用。数据抽象的成功导致在面向对象和纯函数式编程中频繁地将数据隐藏作为设计原则。
在Go语言中,可以通过接口实现抽象。接口定义了一组必须实现的方法,以满足接口的要求。
例如,考虑以下接口:
```go
type iStudent interface{
calcPercentage() float64
}
```
该接口定义了一个名为`calcPercentage()`的方法,用于计算学生的百分比。
## 继承(Inheritance)
在Go语言中,不支持传统意义上的继承。相反,Go使用组合来实现类似的功能。一个结构体可以嵌入另一个结构体,从而继承其字段和方法。
例如,考虑以下结构体:
```go
type Person struct {
name string
age uint64
}
type Student struct {
Person
studentID uint64
}
```
在这里,`Student`结构体嵌入了`Person`结构体,这意味着它继承了`Person`的`name`字段。`Student`结构体还有自己的`studentID`字段。现在,任何对`Person`操作的方法也可以对`Student`操作,因为`Student`是一个带有额外字段的`Person`。
## 多态(Polymorphism)
多态是指调用代码可以独立于所支持层次结构中的类(父类或其子类)而运行的情况。
在Go语言中,可以通过使用接口来实现多态。由于Go是一种静态类型的语言,变量的类型必须在编译时已知。然而,通过使用接口,可以编写可以在满足相同接口要求的不同类型上操作的代码,而无需在编译时知道其具体类型。
例如,考虑以下代码:
```go
func PrintStudentDetails(s *Student) {
fmt.Println("Student Name:", s.GetName())
fmt.Println("Student Age:", s.GetAge())
fmt.Println("Percentage",s.CalcPercentage())
}
```
现在,我们可以将任何实现了`PrintStudentDetails()`方法的类型作为参数调用该函数,比如`Student`类型,它会显示该学生的详细信息。这就是多态的实现。
完整的面向对象编程在Go中的示例:
```go
package main
import "fmt"
type Shape interface {
Area() float64
}
type Triangle struct {
Base, Height float64
}
func (t Triangle) Area() float64 {
return 0.5 * t.Base * t.Height
}
type Square struct {
Side float64
}
func (s Square) Area() float64 {
return s.Side * s.Side
}
type Rectangle struct {
Length, Breadth float64
}
func (r Rectangle) Area() float64 {
return r.Length * r.Breadth
}
func printArea(s Shape) {
fmt.Println("Area of shape is : ", s.Area())
}
func main() {
t := Triangle{Base: 10, Height: 20}
printArea(t) //Area of shape is : 100
s := Square{Side: 20}
printArea(s) //Area of shape is : 400
r := Rectangle{Length: 20, Breadth: 10}
printArea(r) //Area of shape is : 200
}
```
有疑问加站长微信联系(非本文作者)