厚土Go学习笔记 | 28. go语言没有类 却可以在结构体或任意类型定义方法

在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 的输出 vv.Abs() 就和 Before 不同了。

在一些多个点共同对一个点享有修改权的场景中,指针类型很有用。

原文发布于微信公众号 - Golang语言社区(Golangweb)

原文发表时间:2017-12-21

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏恰童鞋骚年

.NET中那些所谓的新语法之二:匿名类、匿名方法与扩展方法

开篇:在上一篇中,我们了解了自动属性、隐式类型、自动初始化器等所谓的新语法,这一篇我们继续征程,看看匿名类、匿名方法以及常用的扩展方法。虽然,都是很常见的东西,...

12630
来自专栏个人随笔

深入.NET数据类型(2)

一.装箱和拆箱   将值类型转换为引用类型的过程称为装箱,反之称为拆箱 ? ?   实际开发尽量避免 装/拆 箱     原因:       装/拆 箱都会降低...

28750
来自专栏风口上的猪的文章

.NET面试题系列[6] - 反射

在面试中,通常会考察反射的定义(操作元数据),可以用反射做什么(获得程序集及其各个部件),反射有什么使用场景(ORM,序列化,反序列化,值类型比较等)。如果答得...

14120
来自专栏GreenLeaves

C#核编之格式化编程

一、格式化控制台输入输出     1、 在前面的随笔中,会经常看到诸如{0},{1}之类的标记嵌入在字符串变量中。.NET引入一种字符串格式化的新风格。与C的p...

206100
来自专栏蘑菇先生的技术笔记

探索c#之递归APS和CPS

30870
来自专栏性能与架构

认识一下 Java 11

可能很多人现在 Java8 的新特性还没用熟呢,Java 11 就已经来了,下面一起来看下 Java 11 的几个新特性:

14420
来自专栏逸鹏说道

Python3 与 C# 基础语法对比(Function专栏)

汇总系列:https://www.cnblogs.com/dunitian/p/4822808.html#ai

9630
来自专栏vue

委托初级篇——lambda表达式的推导

 public delegate void ConsoleWriteStr(string name,DateTime now);

16820
来自专栏zaking's

用js来实现那些数据结构10(集合02-集合的操作)

  前一篇文章我们一起实现了自定义的set集合类。那么这一篇我们来给set类增加一些操作方法。那么在开始之前,还是有必要解释一下集合的操作有哪些。便于我们更快速...

33260
来自专栏blackheart的专栏

[C#6] 4-string 插值

0. 目录 C#6 新增特性目录 1. 老版本的代码 1 internal class Person 2 { 3 public string Na...

21060

扫码关注云+社区

领取腾讯云代金券