GO语言-07派生类型:结构体和方法(下)

初心是记录和总结,自己学习Go语言的历程。如果能帮助到你,这是我的荣幸。

属于一个类型的方法

这里出现一个新的概念:方法。在go语言中,方法和函数还是有区别的:

  • 函数是传入特定值或直接调用后输出结果或直接执行某种操作
  • 而方法是作用于特定类型变量的函数

举个清晰的例子,在我的上一篇文章中,写了一个例子:描述的是给person这个自定义类型创建一个setter方法,使得它可以通过setter对成员进行赋值。当时我们的操作是这样的:

func main() {
    type person struct {
       name    string //描述姓名
       age     int    //描述年龄
       setNameAndAge func(p *person, name string, age int)
    }

    var people person //定义person类型的变量为people

    people.setNameAndAge = func(p *person, name_ string, age_ int) {
       p.name = name_
       p.age = age_
       fmt.Printf("%p\n", p) //0xc0000543c0
    }

    people.setNameAndAge(&people, "张三", 15)
    fmt.Printf("%p\n", &people) //0xc0000543c0
    fmt.Println(people) //{张三 15 0xa4fc80}
}

这样的操作,需要我们在结构体中定义好变量之后,还要为变量初始化:赋值一个方法。比较麻烦不说,当我们输出该变量的时候还会发现方法也被输出了。那前面我们说:方法作用于特定类型变量的函数,我们来看看方法是怎么解决这一类事情的:结构体的方法

方法的语法

   func (接收变量形参名称 接收类型) 方法名(参数列表)(返回值列表){
   
   }

可以看出它和函数的语法很像,唯一区别的地方在于(接收变量形参名称 接收类型),这里就是表明了什么类型可以接收(或调用)该方法。也就是说当我们这样写的时候,就可以通过变量名.方法名调用了方法了!

   func (p person) 方法名(参数列表)(返回值列表){
   
   }
   
   var people person
   people.方法名

使用方法绑定给struct类型

介绍完毕后,我们来实现通过方法实现:给person这个自定义类型创建一个setter方法。

package main

import "fmt"

type person struct {
   name    string //描述姓名
   age     int    //描述年龄
}

func main() {

   // 定义
   var people person

   people.setter("程云来", 18)
   fmt.Println(people) //{程云来 18}
}

func (people *person) setter(name string, age int) {
   people.name = name
   people.age = age
}

(people *person)是将操作的地址进行传递,保证了操作的是同一块内存地址的内容,而不使用指针的方式,一般都是值传递。而变量一般是有作用域范围的,每个学过编程语言的都会有概念,值传递的作用域范围一般在方法中都是方法内局部使用的,如果不将值作为范围值传递回去用参数接收,它在方法执行完毕后就会被垃圾回收


既然我们上面都说了指针作用域范围以及值和地址传递的概念,下面我们通过函数以及方法了解一下这些概念的使用。

指针传递和值传递

我们通过一个例子来感受一下。定义int变量num,分别调用usePoint函数以及notUsePoint两个函数,这两个函数的区别一个是指针形参另一个是类型形参
函数调用结束后,我们会发现:

  • 使用指针的函数改变了原num的值
  • 使用类型形参的函数没有改变num的值
package main

import "fmt"

func main() {
   num := 0
   notUsePoint(num)
   fmt.Println(num) //0
   usePoint(&num)
   fmt.Println(num) //10
}

func usePoint(num *int) {
   *num = 10
}

func notUsePoint(num int) {
   num = 10
}

那么类似的,上面的例子改用方法其实是一样的。但是! 方法有个奇妙的用法并且和函数不同的,记住就行:

  • 接收者为指针类型值类型的方法,指针类型和值类型的变量均可相互调用。但是函数不行!

回到我们之前的例子:

func main(){
   people.setter("程云来", 18)
   fmt.Println(people)
}

func (people *person) setter(name string, age int) {
   people.name = name
   people.age = age
}

发现一个关键要点没!我的方法接受者是指针类型,而值类型变量可以直接调用,但是操作的时候是将值类型变量的地址传递过来了。但是函数形参和传递参数类型不一致的时候就会报错。

可以理解方法中的接受者是方法定义的规则,按照我的规则,我是操作值还是操作地址。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值