普通函数定义


golang函数基本组成:关键字func函数名参数列表返回值函数体返回语句

• 语法如下

func 函数名(参数列表) (返回值列表) {
// 函数体
}

• 示例如下

package main

import "fmt"
import
"errors" func Add(a int, b int) (ret int, err error) {
if a < || b < { // 假定只支持两个非负数字的加法
err = errors.New("Should be non-negative numbers!")
return
}
return a + b, nil
} func main() {
fmt.Println(Add(-, ))
fmt.Println(Add(, ))
}

• 注意事项

▪ golang函数支持多返回值,返回时没有被明确赋值的返回值将被设置为默认值

▪ golang函数返回值可以命名,但不是强制规则,给返回值命名可以让让代码更清晰,可读性更强,同时也可以用于文档。

▪ golang函数定义中左花括号的位置被强制规范,如果左花括号放置不规范,golang编译器报告编译错误:syntax error: unexpected semicolon or newline before {。

▪ golang函数定义中如果参数列表中若干个相邻参数的类型相同,则可以在参数列表中省略前面的变量类型声明。

func Add(a, b int) (ret int, err error) {
    // 函数体
}

▪ golang函数定义中如果返回值列表中若干个相邻返回值的类型相同,则可以在返回值列表中省略前面的变量类型声明。

func Add(a, b int) (ret, err int) {
    // 函数体
}

▪ golang函数定义中如果返回值只有一个,那么返回值列表可以简略表示。

func Add(a, b int) int {
    // 函数体
}

固定类型的不定参数函数定义


• 语法如下

func 函数名(args ...Type) (返回值列表) {
    // 函数体
}

形如args ...Type的用法只能作为函数的参数类型,并且必须是最后一个参数

形如args ...Type的用法是一个语法糖(syntactic sugar),即对语言功能没有影响,但可以增加程序的可读性。

形如args ...Type的用法从内部实现上说,其本质上是一个数组切片,即[]type,因此可以用for循环来获得每个传入的参数。

• 示例如下

package main

import "fmt"

func myfunc(prefix string, args ...int) {
fmt.Print(prefix, " : ") for _, arg := range args {
fmt.Print(arg, " ")
} fmt.Println()
} func main() {
myfunc("data1", , , , )
myfunc("data2", , , , , )
}

• 不定参数函数互相调用

▶ 原样传递

▪ 语法如下

func myfunc1(args ...Type) {
// 函数体
} func myfunc2(args ...Type) {
...
myfunc1(args...)
...
}

▪ 示例如下

package main

import "fmt"

func myfunc1(args ...int) {
fmt.Println("myfunc1() invoked") for _, arg := range args {
fmt.Print(arg, " ")
} fmt.Println()
} func myfunc2(args ...int) {
fmt.Println("myfunc2() invoked")
myfunc1(args...)
} func main() {
myfunc2(, , , )
}

▶ 片段传递

▪ 语法如下

func myfunc1(args ...Type) {
// 函数体
} func myfunc2(args ...Type) {
...
myfunc1(args[m:n]...)
...
}

▪ 示例如下

package main

import "fmt"

func myfunc1(args ...int) {
fmt.Println("myfunc1() invoked") for _, arg := range args {
fmt.Print(arg, " ")
} fmt.Println()
} func myfunc2(args ...int) {
fmt.Println("myfunc2() invoked")
myfunc1(args[:]...)
} func main() {
myfunc2(, , , )
}

任意类型的不定参数函数定义


• 语法如下

func 函数名(args ...interface{}) (返回值列表) {
// 函数体
}

形如args ...interface{}的用法只能作为函数的参数类型,并且必须是最后一个参数。

• 示例如下

package main

import "fmt"

func MyPrintf(info string, args ...interface{}) {
fmt.Println(info)
for _, arg := range args {
switch arg.(type) {
case int:
fmt.Println(arg, "is an int value.")
case string:
fmt.Println(arg, "is a string value.")
case int64:
fmt.Println(arg, "is an int64 value.")
default:
fmt.Println(arg, "is an unknown type.")
}
}
} func main() {
var v1 int =
var v2 int64 =
var v3 string = "hello"
var v4 float32 = 1.234 MyPrintf("MyPrintf", v1, v2, v3, v4)
}

匿名函数与闭包


• 匿名函数

 匿名函数是指不需要定义函数名的一种函数实现方式,函数可以像普通变量一样被传递或使用。

▶ 匿名函数赋值给变量

▪ 语法如下

f := func(参数列表) (返回值列表) {
// 函数体
}

▪ 示例如下

package main

import "fmt"

func main() {
f := func(x, y int) int {
return x + y
} fmt.Println("Result =", f(, ))
}

▶ 匿名函数直接调用

▪ 语法如下

 匿名函数定义的花括号后直接跟参数列表表示函数调用。

func(参数列表) (返回值列表) {
// 函数体
} (参数列表)

▪ 示例如下

package main

import "fmt"

func main() {
fmt.Println("Result =",
func(x, y int) int {
return x + y
}(, ))
}

• 闭包

▶ 闭包概念

 闭包是可以包含自由变量(未绑定到特定对象)的代码块。这些自由变量不在该代码块内或任何全局上下文中定义,而是在定义代码块的环境中定义。要执行的代码块为自由变量提供绑定的计算环境,即作用域。

 自由变量包含在代码块中,这些自由变量以及其引用的对象将不被释放。闭包的实现确保只要闭包还被使用,那么被闭包引用的变量会一直存在。

 golang的匿名函数就是一个闭包,闭包函数可以存储到变量中作为参数传递给其他函数,最重要的是能够被函数动态创建和返回。

 自由变量的值将被隔离,在闭包外不可修改,只有内部匿名函数才能访问,而无法通过其他途径访问,因此确保了变量的安全性。

▶ 闭包使用

▪ 语法如下

c := func() func(参数类型列表) 返回值类型列表 {
// 自由变量定义
...
return func(参数列表) 返回值列表 {
// 函数定义
}
}()

▪ 示例如下

package main

import (
"fmt"
) func main() {
result := // 闭包定义
c := func() func(int) int {
i :=
return func(j int) int {
i +=
fmt.Printf("i = %d, j = %d\n", i, j)
return i + j
}
}() result = c()
fmt.Printf("i + j= %d\n", result) result = c()
fmt.Printf("i + j= %d\n", result) result = c()
fmt.Printf("i + j= %d\n", result)
}

main函数


 main函数是golang可执行程序的执行起点。

 main函数不能带有参数,命令行传入的参数在os.Args变量中保存。

package main

import (
"fmt"
"os"
) func main() {
for i, v := range os.Args {
fmt.Printf("os.Args[%d] = %s\n", i, v)
}
}

[golang note] 函数定义的更多相关文章

  1. golang笔记——函数与方法

    如果你遇到没有函数体的函数声明,表示该函数不是以Go实现的. package math func Sin(x float64) float //implemented in assembly lang ...

  2. golang的函数

    在golang中, 函数是第一类值(first-class object), 即函数可以赋值与被赋值. 换言之, 函数也可以作为ReceiverType, 定义自己的method. 实例: http. ...

  3. GO语言的进阶之路-Golang高级数据结构定义

    GO语言的进阶之路-Golang高级数据结构定义 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 我们之前学习过Golang的基本数据类型,字符串和byte,以及rune也有所了解, ...

  4. Python学习札记(十一) Function2 函数定义

    参考:定义函数 Note: 先看一段代码实例(Barefoot topo.py): def read_topo(): nb_hosts = 0 nb_switches = 0 links = [] w ...

  5. 5.1 Go函数定义

    1 Go函数定义 Go函数是指:一段具有独立功能的代码,然后可以在程序中其他地方多次调用. Go分为自定义函数,系统函数. 函数可以将一个大的工作拆解成小的任务. 函数对用户隐藏了细节. Golang ...

  6. golang:函数总结

    golang保留的函数 init(), main()是golang的保留函数,有如下特点: main() 只能用在main包中,仅可定义一个,init() 可定义任意包,可重复定义,建议只定义一个 两 ...

  7. 深入理解javascript函数定义与函数作用域

    最近在学习javascript的函数,函数是javascript的一等对象,想要学好javascript,就必须深刻理解函数.本人把思路整理成文章,一是为了加深自己函数的理解,二是给读者提供学习的途径 ...

  8. JavaScript函数定义和调用 变量作用域

     本文是笔者在看廖雪峰老师JavaScript教程时的个人总结   JavaScript中函数定义可以是这样的格式 function 函数名(参数) {     函数体 } 也可以是这样的格式     ...

  9. Scala深入浅出实战经典-----002Scala函数定义、流程控制、异常处理入门实战

    002-Scala函数定义.流程控制.异常处理入门实战 Scala函数定义 语句结束无分号 定义无参函数 def 函数名称(参数名称:参数类型)[:Unit=]{ 函数体 } 老师的代码 我的实际代码 ...

随机推荐

  1. TensorFlow基础笔记(8) TensorFlow简单人脸识别

    数据材料 这是一个小型的人脸数据库,一共有40个人,每个人有10张照片作为样本数据.这些图片都是黑白照片,意味着这些图片都只有灰度0-255,没有rgb三通道.于是我们需要对这张大图片切分成一个个的小 ...

  2. YII2与Thinkphp整合阿里云OSS

    前言: 如果上传的文件都和网站程序源代码放在一起:那是有相当多的弊端的: 1:静态文件会占用大量带宽: 2:服务器的成本略高: 常规的做法是把php源代码放到一台服务器上:图片等静态文件放在另一台服务 ...

  3. Linux性能调优、Linux集群与存储等

    http://freeloda.blog.51cto.com/    51cto

  4. XML基础知识-->Spring配置

    XML的特殊字符 XML中共有5个特殊的字符,分别是:&<>“’.如果配置文件中的注入值包括这些特殊字符,就需要进行特别处理.有两种解决方法:其一,采用本例中的<![CDAT ...

  5. C# 直接调用非托管代码的方法

    C# 代码有以下两种可以直接调用非托管代码的方法: 直接调用从 DLL 导出的函数. 调用 COM 对象上的接口方法. 对于这两种技术,都必须向 C# 编译器提供非托管函数的声明,并且还可能需要向 C ...

  6. AWS CLI 【S3】

    1.创建一个桶&删除一个桶 root@syavingc:~# aws s3 mb s3://syavingc #创建一个桶 make_bucket: syavingc root@syaving ...

  7. 第十三篇:带缓冲的IO( 标准IO库 )

    前言 在之前,学习了 read write 这样的不带缓冲IO函数. 而本文将讲解标准IO库中,带缓冲的IO函数. 为什么要有带缓冲IO函数 标准库提供的带缓冲IO函数是为了减少 read 和 wri ...

  8. D3D中的渲染状态简介

    1). 设置着色模式: SetRenderState(D3DRS_SHADEMODE, D3DSHADE_FLAT) //设置平面着色模式 SetRenderState(D3DRS_SHADEMODE ...

  9. Android遍历SqlLite cursor对象:

    //1. Cursor c =...; for(c.moveToFirst(); ! c.isAfterLast(); c.moveToNext()){ //c… } //2. Cursor curs ...

  10. 170407、java基于nio工作方式的socket通信

    客户端代码: /** * */ package com.bobohe.nio; import java.io.BufferedReader; import java.io.IOException; i ...