Go语言之函数方法

在Go语言中,函数和方法不太一样,有明确的概念区分。在其他语言中,比如Java,一般来说,函数就是方法,方法就是函数。但是在Go语言中,函数是指不属于任何结构体、类型的方法。也就是说,函数是没有接收者的;而方法是有接收者的。我们说的方法要么是属于一个结构体的,要么属于一个新定义的类型的。

函数

函数和方法,虽然概念不同,但是定义非常相似。函数的定义声明没有接收者,所以我们直接在Go文件里、Go包之下定义声明即可。

func main() {
    sum := add(1, 2)
    fmt.Println(sum)
}

func add(a, b int) int {
    return a + b
}

例子中,我们定义了add就是一个函数,它的函数签名是func add(a, b int) int。没有接收者,直接定义在Go的一个包之下,可以直接调用,比如例子中的main函数调用了add函数。

例子中的这个函数名称是小写开头的add,所以它的作用域只属于所声明的包内使用,不能被其他包使用。如果我们把函数名以大写字母开头,该函数的作用域就大了,可以被其他包调用。这也是Go语言中大小写的用处。比如Java中,就有专门的关键字来声明作用域privateprotectpublic等。

/*
 提供的常用库,有一些常用的方法,方便使用
*/
package lib

// 一个加法实现
// 返回a+b的值
func Add(a, b int) int {
    return a + b
}

如上例子中定义的Add方法就可以被其他包调用。

方法

方法的声明和函数类似,他们的区别是:方法在定义的时候,会在func和方法名之间增加一个参数,这个参数就是接收者,这样我们定义的这个方法就和接收者绑定在了一起,称之为这个接收者的方法。

type person struct {
    name string
}

func (p person) String() string{
    return "the person name is "+p.name
}

留意例子中,func和方法名之间增加的参数(p person),这个就是接收者。现在我们说,类型person有了一个String方法,现在我们看下如何使用它。

func main() {
    p:=person{name:"张三"}
    fmt.Println(p.String())
}

调用的方法非常简单,使用类型的变量进行调用即可,类型变量和方法之前是一个.操作符,表示要调用这个类型变量的某个方法的意思。

Go语言里有两种类型的接收者:值接收者和指针接收者。上面就是使用值类型接收者的示例。

使用值类型接收者定义的方法,它调用的其实是值接收者的一个副本,所以对该值的任何操作,不会影响原来的类型变量。

func main() {
    p:=person{name:"张三"}
    p.modify() //值接收者,修改无效
    fmt.Println(p.String())
}

type person struct {
    name string
}

func (p person) String() string{
    return "the person name is "+p.name
}

func (p person) modify(){
    p.name = "李四"
}

以上的例子,打印出来的值还是张三,对其进行的修改无效。如果我们使用一个指针作为接收者,那么就会其作用了。因为指针接收者传递的是一个指向原值指针的副本,它指向的还是原来类型的值,所以修改时,同时也会影响原来类型变量的值。

func main() {
    p:=person{name:"张三"}
    p.modify() //指针接收者,修改有效
    fmt.Println(p.String())
}

type person struct {
    name string
}

func (p person) String() string{
    return "the person name is "+p.name
}

func (p *person) modify(){
    p.name = "李四"
}

只需要改动一下,变成指针的接收者,就可以完成修改。

在调用方法的时候,传递的接收者本质上都是副本,只不过一个是这个值副本,一个是指向这个值指针的副本。指针具有指向原有值的特性,所以修改了指针指向的值,也就修改了原有的值。我们可以简单地理解为值接收者使用的是值的副本来调用方法,而指针接收者使用实际的值来调用方法。

有没有发现,在上面的例子中,我们在调用指针接收者方法的时候,使用的也是一个值的变量,并不是一个指针。如果我们使用下面的方法也是可以的。

p:=person{name:"张三"}
(&p).modify() //指针接收者,修改有效

如果我们没有这么强制使用指针进行调用,Go的编译器会自动帮我们取指针,以满足接收者的要求。

同样的,如果是一个值接收者的方法,使用指针也是可以调用的。Go编译器会自动解引用,以满足接收者的要求。比如例子中定义的String()方法,也可以这么调用:

p:=person{name:"张三"}
fmt.Println((&p).String())

总之,方法的调用,既可以使用值,也可以使用指针。我们不必要严格的遵守这些,Go语言编译器会帮我们进行自动转义的,这大大方便了我们开发者。

不管是使用值接收者,还是指针接收者,一定要搞清楚类型的本质:对类型进行操作的时候,是要改变当前值,还是要创建一个新值进行返回?这些就可以决定我们是采用值传递,还是指针传递。

多值返回

Go语言支持函数方法的多值返回,也就说我们定义的函数方法可以返回多个值。比如标准库里的很多方法,都是返回两个值:第一个是函数需要返回的值,第二个是出错时返回的错误信息。这样的好处是,我们的出错异常信息再也不用像Java一样,需要使用Exception这么重的方式表示了,非常简洁。

func main() {
    file, err := os.Open("/usr/tmp")
    if err != nil {
        log.Fatal(err)
        return
    }
    fmt.Println(file)
}

返回的值如果我们不想使用,可以使用_进行忽略。

file, _ := os.Open("/usr/tmp")

多个值返回的定义也非常简单,看个例子。

func add(a, b int) (int, error) {
    return a + b, nil
}

函数方法声明定义的时候,采用逗号分割,因为是多个返回,还要用括号括起来。返回的值还是使用return关键字,以逗号分割,和返回声明的顺序一致。

可变参数

函数方法的参数,可以是任意多个,这种我们称之为可以变参数。比如我们常用的fmt.Println()这类函数,可以接收一个可变的参数。

func main() {
    fmt.Println("1","2","3")
}

可以变参数,可以是任意多个。我们自己也可以定义可以变参数,可变参数的定义,在类型前加上省略号...即可。

func main() {
    print("1","2","3")
}

func print (a ...interface{}){
    for _,v:=range a{
        fmt.Print(v)
    }
    fmt.Println()
}

例子中我们自己定义了一个接受可变参数的函数,效果和fmt.Println()一样。

可变参数本质上是一个数组,所以我们像使用数组一样使用它,比如例子中的 for range循环。

时间: 2024-08-28 03:19:37

Go语言之函数方法的相关文章

C/C++语言参数传递----函数/方法 参数的指针引用传递

int m_value = 1; void func(int *p) { p = &m_value; } int main(int argc, char *argv[]) { int n = 2; int *pn = &n; cout << *pn << endl; func(pn); cout << *pn <<endl; return 0; } 看一下输出结果 22 -------其实上面这些例子,看一百次,我个人觉得,也看不出实际意义

[C/C++基础] C语言常用函数sprintf和snprintf的使用方法

Sprintf 函数声明:int sprintf(char *buffer, const char *format [, argument1, argument2, -]) 用途:将一段数据写入以地址buffer开始的字符串缓冲区 所属库文件: <stdio.h> 参数:(1)buffer,将要写入数据的起始地址:(2)format,写入数据的格式:(3)argument:要写的数据,可以是任何格式的. 返回值:实际写入的字符串长度 说明:此函数需要注意缓冲区buffer溢出,要为写入的arg

[C/C++基础] C语言常用函数strlen的使用方法

函数声明:extern unsigned int strlen(char *s); 所属函数库:<string.h> 功能:返回s所指的字符串的长度,其中字符串必须以'\0'结尾 参数:s为字符串的初始地址 使用举例: 代码如下 编译运行结果 说明: 函数strlen比较容易理解,其功能和sizeof很容易混淆.其中sizeof指的是字符串声明后占用的内存长度,它就是一个操作符,不是函数:而strlen则是一个函数,它从第一个字节开始往后数,直到遇见了'\0',则停止. [C/C++基础] C

[C/C++基础] C语言常用函数memset的使用方法

函数声明:void *memset(void *s, int ch, size_t n); 用途:为一段内存的每一个字节都赋予ch所代表的值,该值采用ASCII编码. 所属函数库:<memory.h> 或者 <string.h> 参数:(1)s,开始内存的地址:(2)ch和n,从地址s开始,在之后的n字节长度内,把每一个字节的值都赋值为n. 使用举例: 代码如下 编译运行结果 说明: 该函数最常用的用途就是将一段新分配的内存初始化为0.例如我们代码的第9-10行. 需要注意的是,函

C语言基础 函数--思想的体现

1.C语言由函数组成,main函数又称主函数,是程序的入口. 1.1函数定义:    1.1.1    格式:        返回值类型 函数名(形式参数列表)        {            函数体        }     1.1.2    函数名:  不允许相同 1.1.3    参数注意点:        1.形式参数: 定义函数时函数名后面括号中的参数,简称形参        2.实际参数: 调用函数时传入的具体数据,简称实参        3.实参个数必须等于形参个数    

VC开发多语言界面 多种方法(很简单) 有源码

(需源码先留邮箱)先上图 1.通过遍历 得到所有控件ID号与TEXT,得到一个中文语言配置文件 void CVV_485Dlg::getCaptionForWindow() //做程序时用,其它时间不用 { //枚举对话框中所有组件 CWnd *pCtrl = GetWindow(GW_CHILD); while(pCtrl!=NULL) { UINT ctrlID = pCtrl->GetDlgCtrlID(); // setControlCaption(pCtrl,ctrlID); CStr

【C语言】函数指针与回调函数

在C语言中:指针是C语言的特色,有着各种各样的指针,普通的变量指针,常量指针,数组指针,指针数组,函数指针,指针函数.我们就讲一下函数指针与回调函数吧 首先关于函数指针,其实很简单. 对于一个函数指针来说,顾名思义,就是一个指向函数的指针,需要知道的是,对于指针而言,他总是存储一块地址,地址里面有着一个,一组,或者一块数据,在函数中,函数的存储是放在代码段的,每个函数都有着一个函数首地址,调用了这个地址相当于调用的这个函数. 具体的可以观看我的这篇博客,其中就通过在内存阶段改变栈帧返回值,成功的

C语言的函数返回值

一:背景 谈到C语言的函数返回值,可能会感觉很亲切,不就是一个函数返回值嘛,当初学C语言的时候早就学过了很easy嘛,我曾经也是这么想的.后来要上研究生了,研究生阶段搞得就是C,所以又重新开始学习C,学习C的过程中遇到了很多问题,在此博客中一一记录.实际过程中遇到的第一个问题自然就是函数返回值了.如果有人问你在一个函数中声明一个字符串数组,最后再return这个数组.这可以实现嘛?如果是问我我可能会毫不犹豫的说OK.那事实呢?由此本文诞生了...... 二:问题 先看几个实际的例子: #incl

Android C语言_init函数和constructor属性及.init/.init_array节探索

本篇文章主要介绍了"Android C语言_init函数和constructor属性及.init/.init_array节探索",主要涉及到Android C语言_init函数和constructor属性及.init/.init_array节探索方面的内容,对于Android C语言_init函数和constructor属性及.init/.init_array节探索感兴趣的同学可以参考一下. 了解C语言的程序猿都知道有两种方法可以让一部分代码在so或可执行文件被加载的时候先于其它任何函