下面由golang教程栏目给大家详解 Go 中方法与函数的区别,希望对需要的朋友有所帮助!

这篇文章将介绍Go中函数和方法之间的主要区别,以及如何最佳使用。



Go中广泛使用了函数和方法来提供抽象,使我们的程序更易于阅读和推理。从表面上来看,函数和方法看起来都相似的,但是存在一些重要的语义差异,这些差异可能会极大地影响代码的可读性。

语法
声明语法
通过指定参数的类型、返回值和函数主体来声明函数:

1

2

3

4

5

6

7

8

9

10

11

12

type Person struct {

  Name string

  Age int

}

 

// 这个函数返回一个新的对象`Person`

func NewPerson(name string, age int) *Person {

  return &Person{

    Name: name,

    Age: age,

  }

}

另一方面,通过额外指定“接收者”(在OOP术语中将是该方法所属的“类 class”)来声明方法:

1

2

3

4

// `Person`指针类型接收者的`isAdult方法

func (p *Person) isAdult() bool {

  return p.Age > 18

}

在上面的方法声明中,我们在*Person类型上声明了isAdult方法 。

执行语法
函数调用使用独立的参数,方法调用使用接收者类型。

1

2

3

4

p := NewPerson("John", 21)

 

fmt.Println(p.isAdult())

// true

互换性
函数和方法 在 理论上可以互换。例如,我们可以将isAdult方法转换为函数,并将NewPerson函数作为方法:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

type PersonFactory struct {}

 

// NewPerson现在是PersonFactory结构的方法

func (p *PersonFactory) NewPerson(name string, age int) *Person {

  return &Person{

    Name: name,

    Age: age,

  }

}

 

// 现在,isAdult是一个函数,在该函数中,我们将`Person`作为参数而不是接收者进行传递

func isAdult(p *Person) bool {

  return p.Age > 18

}

在这种情况下,执行语法看起来有些奇怪:

1

2

3

4

5

6

factory := &PersonFactory{}

 

p := factory.NewPerson("John", 21)

 

fmt.Println(isAdult(p))

// true

上面的代码看起来比需要的要复杂得多。这向我们表明,方法和函数的差异主要是语法上的差异,应该根据场景使用适当的抽象。

用例
让我们看一下Go应用程序中遇到的一些常见用例,以及适用于每个应用程序的适当抽象(函数或方法):

方法链
方法的一个非常有用的特性是能够将它们链接在一起,同时仍保持代码的清洁。让我们以设置Person使用链接的某些属性为例 :

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

type Person struct {

    Name string

    Age  int

}

 

func (p *Person) withName(name string) *Person {

    p.Name = name

    return p

}

 

func (p *Person) withAge(age int) *Person {

    p.Age = age

    return p

}

 

func main() {

    p := &Person{}

 

    p = p.withName("John").withAge(21)

 

  fmt.Println(*p)

  // {John 21}

}

如果我们将函数用于同一件事,那将看起来非常可怕:

1

p = withName(withAge(p, 18), "John")

有状态与无状态执行
在 可互换性 示例中,我们看到了使用PersonFactory对象来创建Person的新实例。事实证明,这是一种反模式,应避免使用。

最好使用NewPerson以前声明的函数之类的函数进行无状态执行。

这里的“无状态”是指始终为同一输入返回相同输出的任何代码

推论是,如果您发现函数读取并修改了许多特定类型的值,则它可能应该定义成该类型的方法。

语义
语义是指代码的阅读方式。如果你用口语大声朗读代码,那么哪个更有意义?

我们来看看 isAdult 的函数和方法实现

1

2

3

4

5

6

7

customer := NewPerson("John", 21)

 

// Method

customer.isAdult()

 

// Function

isAdult(customer)

在这里 customer.isAdult() 对于询问「客户是否是成年人?」的理解要比isAdult(customer) 好得多。此外,当你问「x 是成年人吗?」时,总是会在 x 的上下文中被问到。

结论
尽管我们讨论了 Go 中函数和方法的一些关键区别和用例,但总会有例外!重要的是,不要将任何这些规则都作为基本原则。

最后,函数和方法之间的区别在于结果代码的读取方式。如果您或您的团队认为一种方法读起来比另一种更好,那么这就是正确的抽象!