普通函数定义


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. svn & git 问题汇总

    svn: warning: W150002: '/Users/piercalex/svn/azzz/1' is already under version control svn: E200009: ...

  2. ubuntu 安装dlib 出现dlib.so: undefined symbol: png_set_longjmp_fn

    参考网上的教程安装dlib 安装教程1 sudo apt-get install libboost-python-dev cmake sudo pip install dlib 安装教程2ubuntu ...

  3. 【UVa】Wavio Sequence(dp)

    http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&p ...

  4. 《算法导论》— Chapter 12 二叉查找树

    序 查找树是一种数据结构,它支持多种动态集合操作.包含Search.Minimum.Maximum.PreDecessor.Successor.Insert.Delete等.它既能够用作字典,也能够用 ...

  5. 漫游Kafka设计篇之Producer和Consumer(4)

    Kafka Producer 消息发送 producer直接将数据发送到broker的leader(主节点),不需要在多个节点进行分发.为了帮助producer做到这点,所有的Kafka节点都可以及时 ...

  6. Spring security UserDetailsService autowired注入失败错误

    最近使用spring mvc + spring security 实现登录权限控制的时候,一直不能成功登录,检查过后是dao一直无法注入为null CustomUserDetailConfig.jav ...

  7. Laravel5.1 文件管理

    Laravel提供了一套很好用的文件系统 方便于管理文件夹和文件,支持Amazon S3和Rackspace云存储等驱动. 1 配置 文件系统的配置文件在 config/filesyetems.php ...

  8. Angular2 Observable 可观察对象

    可观察对象支持在应用中的发布者和订阅者之间传递消息.在需要进行事件处理,异步编程和处理多值的时候,可观察对象相对其他技术有显著的优点. 可观察对象是声明式的 —— 也就是说,虽然你定义了一个用于发布值 ...

  9. WPF 隧道路由事件

    阅读本文前,请先了解 冒泡路由事件:http://www.cnblogs.com/andrew-blog/p/WPF_BubbledEvent.html 隧道路由事件的工作方式和冒泡路由事件相同,但方 ...

  10. ajax请求加全局loading , 个别特殊请求不显示loading

    项目中,请求开始前加载loading遮罩层,请求结束关闭遮罩,一般都会加在全局中,但有个别请求不需要加全局loading 的话,这时候就需要对这些请求进行配置 全局加loading: ; functi ...