美文网首页Golang 入门资料+笔记程序员Golang
28. go语言没有类 却可以在结构体或任意类型定义方法

28. go语言没有类 却可以在结构体或任意类型定义方法

作者: 厚土火焱 | 来源:发表于2017-08-18 00:50 被阅读126次

在go语言中没有类。可是,是有方法的。
给结构体定义方法,在对应的 func 和方法名之间,加上方法的接收者就可以了。
比如,我们定义了一个结构体

type Vertex struct {
    X, Y float64
}

希望 Vertex 有一个 abs() 方法,就这样写

func (v *Vertex) Abs() float64 {
    return math.Sqrt(v.X * v.X + v.Y * v.Y)
}

注意结构体的方法接收者是指针时,调用需要在前面加上 & 符号。

(&Vertex{3, 4}).Abs()

这样写有点冗长,在结构的方法有多个的时候,调用也不方便。你可以初始化一个变量,然后再调用。

v := &Vertex{3, 4}
v.Abs()

完整代码看一下

package main

import(
    "fmt"
    "math"
)

type Vertex struct {
    X, Y float64
}

func (v *Vertex) Abs() float64 {
    return math.Sqrt(v.X * v.X + v.Y * v.Y)
}
func main() {
    v := &Vertex{3, 4}
    fmt.Println(v.Abs())
}

这个示例输出的结果是 5
除了结构体,还可以对自己包中的任意类型,定义任意方法。(对来自其他包的类型或基础类型是不能定义方法的。)
比如,你可以创建一个类型

type MyFloat float64

然后给这个 MyFloat 定义方法

func (f MyFloat) Abs() float64 {
    if f < 0 {
        return float64(-f)
    }
    return float64(f)
}

完整示例

package main

import ("fmt"
"math"
)
type MyFloat float64

func (f MyFloat) Abs() float64 {
    if f < 0 {
        return float64(-f)
    }
    return float64(f)
}
func main() {
    f := MyFloat(-math.Sqrt2)
    fmt.Println(f.Abs())
}

运行结果

1.4142135623730951

上面两个例子中,分别实现了两个 Abs() 。一个指针类型,一个值类型。
使用指针是为了避免在每个方法调用时都进行值拷贝(如果类型是大型结构体的话,会更有效率。);其次,指针方法可以修改接收者指向的值。
我们定义一个 Scale 方法,用指针做类型做接收者。会发现 Scale 之后,Vertex 的值发生了改变,并影响以后的运算结果。

package main
import(
    "fmt"
    "math"
)

type Vertex struct {
    X, Y float64
}

func (v *Vertex) Scale(f float64)  {
    v.X = v.X * f
    v.Y = v.Y * f
}
func (v Vertex) Abs() float64  {
    return math.Sqrt(v.X * v.X + v.Y * v.Y)
}
func main() {
    v := Vertex{3, 4}
    fmt.Printf("Before scaling: %+v, ABS: %v\n", v, v.Abs())
    v.Scale(5)
    fmt.Printf("After scaling: %+v, ABS: %v\n", v, v.Abs())
}

运行结果是

Before scaling: {X:3 Y:4}, ABS: 5
After scaling: {X:15 Y:20}, ABS: 25

之所以 Before 和 After 的结果不同,是因为 func (v *Vertex) Scale(f float64) 对 X 和 Y 做了修改,由于使用的是指针类型,所以事实上修改了 Vertex 结构体中 X 和 Y 的值。
然后 After 的输出 v 和 v.Abs() 就和 Before 不同了。
在一些多个点共同对一个点享有修改权的场景中,指针类型很有用。

相关文章

  • go入门(五) 2018-07-19

    方法和接口 方法 Go语言中没有类,但是可以在结构类型上定义方法,实际上可以对包中的任意类型定义任意方法,但是不能...

  • 28. go语言没有类 却可以在结构体或任意类型定义方法

    在go语言中没有类。可是,是有方法的。给结构体定义方法,在对应的 func 和方法名之间,加上方法的接收者就可以了...

  • Golang 方法接受者(receiver of method)

    在go语言中,没有类的概念但是可以给类型(结构体,自定义类型)定义方法。所谓方法就是定义了接受者的函数。接受者定义...

  • 16 Golang结构体详解(二)

    结构体方法和接受者 在go中,没有类的概念但是可以给类型(结构体、自定义类型)定义方法。所谓方法就是定义了接受者的...

  • 09. Go极简教程 结构体的函数扩展

    Go 没有类。然而,仍然可以在结构体类型上定义方法。 结构体的函数扩展可以帮助我们实现类似面向对象的"类的封装" ...

  • 使用Golang完成的第一个项目

    Go语言和PHP的区别(Go语言特色) 没有类的概念,可以用结构体替代。给结构体定义变量,声明方法。 两个特殊的方...

  • Go 语言程序设计——面向对象编程(5)

    结构体 Go 语言中创建自定义结构体最简单的方式是基于 Go 语言的内置类型创建 自定义类型也可以基于结构体创建,...

  • 【技术】Golang进阶(一)

    本篇内容包括: 方法 方法 Go 没有类。然而,仍然可以在结构体类型上定义方法。方法接收者 出现在 func 关键...

  • Go Method

    Go语言同时支持函数和方法,方法是包含接收器的函数,接收器可以是命名类型或结构体类型的值或指针。为特定类型定义的方...

  • 2019-05-22 Go语言学习三 方法和接口

    1、方法 Go 没有类。不过你可以为结构体类型定义方法。方法就是一类带特殊的 **接收者** 参数的函数。方法接收...

网友评论

  • 知识学者::grin: go不该不是很大众,要不我也想学一下。。。

本文标题:28. go语言没有类 却可以在结构体或任意类型定义方法

本文链接:https://www.haomeiwen.com/subject/gtqorxtx.html