go基础之--函数和map
在整理函数之前先整理一下关于指针
指针
普通类型变量存的就是值,也叫值类型。指针类型存的是地址,即指针的值是一个变量的地址。
一个指针指示值所保存的位置,不是所有的值都有地址,但是所有的变量都有。使用指针可以在无序知道
变量名字的情况下,间接读取或更新变量的值。
获取变量的地址,用&,例如:var a int 获取a的地址:&a,&a(a的地址)这个表达式获取一个指向整形变量的指针,它的类型是整形指针(*int),如果值叫做p,我们说p指向x,或者p包含x的地址,p指向的变量写成
*p ,而*p获取变量的值,这个时候*p就是一个变量,所以可以出现在赋值操作符的左边,用于更新变量的值
指针类型的零值是nil
两个指针当且仅当指向同一个变量或者两者都是nil的情况才相等
通过下面小例子进行理解指针:
package main import (
"fmt"
) func test() {
x := 1
// &x 获取的是变量x的地址,并赋值给p,这个时候p就是一个指针
p := &x
// p是指针,所以*p获取的就是变量的值,指针指向的是变量x的值,即*p为1
fmt.Println(*p)
// 这里*p 进行赋值,也就是更改了变量x的值,即实现不知道变量的名字更改变量的值
*p = 2
fmt.Println(x)
} func main() {
test()
}
再看一个关于通过一个函数来修改变量值的问题:
package main import (
"fmt"
) func modify(num int) {
num = 100
} func main() {
a := 10
modify(a)
fmt.Println(a)
}
这个例子是修改变量的值,但是最后打印变量a的值是还是10,所以这里就需要知道,当通过定义的函数modify来修改变量的值时,传入变量a其实会进行一次拷贝,传入的其实是a变量的一个副本,所以当通过
modify修改的时候修改的是副本的值,并没有修改变量a的值。
当我们理解指针的之后,就可以通过指针的的方法来解决上面的这个问题,将代码更改为:
package main import (
"fmt"
) func modify(num *int) {
*num = 100
} func main() {
a := 10
modify(&a)
fmt.Println(a)
}
这里定义modify函数的时候参数设置的是一个指针,所以我们传入参数时,传入的是&a即变量a的地址,而这个地址指向的值是10,虽然这次传入的参数也是进行了传入的指针进行了一次拷贝,但是即使是拷贝了副本指向的值还是10,所以当我们通过指针*num修改值的时候其实就是在修改变量a的值。
内置函数
len: 用于求长度,比如string、array、slice、map、channel
new: 来分配内存,主要来分配值类型,如int、struct。返回的是指针
make: 来分配内存,主要 来分配引 类型, 如chan、map、slice
append: 来追加元素到数组、slice中
panic和recover: 来做错误(这个后续整理)
下面重点整理new和make
new函数
func new(Type) *Type
先看一下官网对这个内置函数的介绍:
内置函数 new 用来分配内存,它的第一个参数是一个类型,不是一个值,它的返回值是一个指向新分配类型零值的指针。这里要特别注意new返回的是一个指针
new函数也是创建变量的一种方式。表达式new(T)创建一个未命名的T类型变量,初始化T类型的零值,并返回其地址(地址类型为*T)
通过下面例子进行理解:
package main import "fmt" func newFunc() {
p := new(int)
fmt.Println(p) //打印是地址
fmt.Println(*p) //int类型的零值为0这里打印0
*p = 2
fmt.Println(*p) //*p已经为其地址指向了一个变量2,所以这里打印为2
} func main() {
newFunc()
}
这里我们要知道new创建的变量和取其地址的普通局部变量没有什么不同,只是语法上的便利
下面是两种方式的例子:
func newInt() *int {
return new(int)
} func newInt2() *int {
var res int
return &res
}
如果我们定义一个指针是不能直接给这个指针赋值的,而是需要先给这个指针分配内存,然后才能赋值
下面例子先不初始化分配内存,直接赋值:
正确的做法是我们需要先通过new初始化,正确代码如下:
package main import (
"fmt"
) func test(){
var p *int
p = new(int)
*p = 10
fmt.Println(*p)
} func main(){
test()
}
make函数
func make(Type, size IntegerType) Type
先看一下官网对这个内置函数的介绍:
内置函数make用来为slice,map或chan类型分配内存或初始化一个对象(这里需要注意:只能是这三种类型)
第一次参数也是一个类型而不是一个值
返回的是类型的引用而不是指针,而且返回值也依赖具体传入的类型
注意:make返回初始化后的(非零)值。
其实在上一篇整理切片slice的时候就用到了make如:
make([]type,len)
当时通过make来初始化slice的时候,第二个参数指定了它的长度,如果吗,没有第三个参数,它的容量和长度相等,当然也可以传入第三个参数来指定不同的容量值,但是注意不能比长度值小
这里提前说一下通过make初始化map的时候,根据size大小来初始化分配内存,不过分配后的map长度为0,如果size被忽略了,会在初始化分配内存的时候分配一个小的内存
关于new和make的一个小结:
new 的作用是初始化一个指向类型的指针 (*T),make的作用是为slice,map或者channel初始化,并且返回引用 T
函数
函数的声明语法:func 函数名 (参数 表) [(返回值 表)] {}
这了要注意第一个花括号必须和func在一行
常见的几种声明函数的方法:
func add(){
}
func add(a int,b int){
}
func add(a int,b int) int{
}
func add(a int, b int)(int,int){
}
func add(a ,b int)(int,int){
}
golang函数的特点:
- 不支持重载,即一个包不能有两个名字一样的函数
- 函数也是一种类型,一个函数可以赋值给变量
- 匿名函数
- 多返回值
演示一些函数的例子:
package main import (
"fmt"
) func add(a, b int) int {
return a + b
} func main() {
c := add //这里把函数名赋值给变量c
fmt.Printf("%p %T", c, add)
sum := c(10, 20) //调用c其实就是在调用add
fmt.Println(sum) }
golang函数还有一个用法例子:
package main import (
"fmt"
) type addFunc func(int, int) int func add(a, b int) int {
return a + b
} func operator(op addFunc, a int, b int) int {
return op(a, b)
} func main() {
c := add
sum := operator(c, 100, 200)
fmt.Println(sum)
}
变量作用域
在函数外面的变量是全局变量
函数内部的变量是局部变量
go中变量的作用域有多种情况:
函数级别的,代码块级别的
通过下面例子理解:
关于函数的可变参数
变长函数被调用的时候可以有可变的参数个数
在参数列表最后的类型名称前使用省略号...可以声明一个变长的函数,
例如:
0个或多个参数
func add(arg...int) int{
}
1个或多个参数
func add(a int,arg...int) int{
}
2个或多个参数
func add(a int,b int,arg...int)int{
}
关于函数参数的传递
不管是值类型还是引用传递,传递给函数的都是变量的副本
注意:map,slice,chan,指针,interface默认以引用方式传递
延迟函数defer的调用
语法上,一个defer语句就是一个普通的函数或者方法调用,在调用之前加上关键字defer。函数和参数表达式会在语句执行时求值,但是无论是正常情况还是执行return语句或者函数执行完毕,以及不正常情况下,如程序发生宕机,实际的调用推迟到包含defer语句的函数结束后才执行,defer语句没有限制使用次数。
defer用途:
- 当函数返回时,执行defer语句,因此可以用来做资源清理
- 多个defer语句,按先进后出的方式执行
- defer语句中的变量,在defer声明时就决定了
先通过一个小例子理解defer:
package main import (
"fmt"
) func testDefer(){
a := 100
fmt.Printf("before defer:a=%d\n",a)
defer fmt.Println(a)
a = 200
fmt.Printf("after defer:a=%d\n",a) } func main(){
testDefer()
}
这里我们可以这样理解当我们执行defer语句的时a=100,这个时候压入到栈中,等程序最后结束的时候才会调用defer语句,所以打印的顺序是最后才打印一个数字100
defer语句经常使用成对的操作,比如打开和关闭,连接和断开,加锁和解锁
下面拿关闭一个打开文件操作为例子,当我们通过os.Open()打开一个文件的时候可以在后面添加defer f.Close() 这样在函数结束时就可以帮我们自动关闭一个打开的文件
Map类型
key-value的数据结构,又叫字典
声明
var map1 map[keytype]valuetype
例子:
var a map[string]string
var a map[string]int
注意:声明是不会分配内存的需要make初始化
初始化的两种方式:
var map[string]string = map[string][string]{"hello","world"}
或:
var a = make(map[string]string,10)
插入和更新
a["hello"] = "world"
查找
val,ok := a["hello"]
遍历
for k,v := range a{
fmt.println(k,v)
}
删除
delete(a,"hello")
这个操作是安全的,及时这个元素不存在也不会报错,如果一个查找失败将返回value类型对应的零值
长度
len(a)
map是引用类型
注意:map中的元素并不是一个变量,所以我们不能对map的元素进行取址操作
go基础之--函数和map的更多相关文章
- Python3基础-特别函数(map filter partial reduces sorted)实例学习
1. 装饰器 关于Python装饰器的讲解,网上一搜有很多资料,有些资料讲的很详细.因此,我不再详述,我会给出一些连接,帮助理解. 探究functools模块wraps装饰器的用途 案例1 impor ...
- python基础——匿名函数
python基础——匿名函数 当我们在传入函数时,有些时候,不需要显式地定义函数,直接传入匿名函数更方便. 在Python中,对匿名函数提供了有限支持.还是以map()函数为例,计算f(x)=x2时 ...
- python基础——filter函数
python基础——filter函数 Python内建的filter()函数用于过滤序列. 和map()类似,filter()也接收一个函数和一个序列.和map()不同的是,filter()把传入的函 ...
- ArcGIS Runtime for Android开发教程V2.0(3)基础篇---Hello World Map
原文地址: ArcGIS Runtime for Android开发教程V2.0(3)基础篇---Hello World Map - ArcGIS_Mobile的专栏 - 博客频道 - CSDN.NE ...
- Go语言基础之函数
Go语言基础之函数 函数是组织好的.可重复使用的.用于执行指定任务的代码块.本文介绍了Go语言中函数的相关内容. 函数 Go语言中支持函数.匿名函数和闭包,并且函数在Go语言中属于“一等公民”. 函数 ...
- 周末班:Python基础之函数进阶
迭代器和生成器 迭代和可迭代 什么是迭代(iteration)? 如果给定一个list或tuple,我们要想访问其中的某个元素,我们可以通过下标来,如果我们想要访问所有的元素,那我们可以用for循环来 ...
- Golang基础之函数
golang基础之函数 1.为什么需要函数? 有些相同的代码可能出现多次,如果不进行封装,那么多次写入到程序中,会造成程序冗余,并且可读性降低 2.什么是函数 为完成某些特定功能的程序指令集合称为函数 ...
- GoLang基础数据类型--->字典(map)详解
GoLang基础数据类型--->字典(map)详解 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 可能大家刚刚接触Golang的小伙伴都会跟我一样,这个map是干嘛的,是 ...
- python基础知识06-函数基础和函数参数
函数基础和函数参数 可迭代对象:序列类型 range . 1.函数的定义 def 函数名(参数): pass return 表达式 ,不能是赋值语句.不写默认返回None.用逗号隔开返回一个元组. 函 ...
随机推荐
- 【python】内置函数总结(一)
1.判断真假的函数:bool()2.Python中所谓的迭代协议就是next方法的对象会前进到下一个结果,在一系列结果的末尾会引发StopIteration异常.在Python中,任何类型的对象都被认 ...
- 手工搭建基于ABP的框架(3) - 登录,权限控制与日志
为了防止不提供原网址的转载,特在这里加上原文链接: http://www.cnblogs.com/skabyy/p/7695258.html 本篇将实现登录.权限控制.日志配置与审计日志的功能.首先我 ...
- Liunx vi编辑器一些指令
最近几天学习了Liunx vi编辑器 的使用,感觉还比较容易.总结的一点心得: vi分为3个模式,命令模式,尾行模式,编辑模式. 1. 命令模式 与 编辑模式切换 a:光标向后移动一位进入编辑模式 i ...
- JavaScript的兼容
兼容总结 如果两个都是属性,用逻辑 || 做兼容 如果有一个是方法,用三元做兼容 如果多个属性或方法,封装函数做兼容 获取class属性值的兼容 function getClass (obj){ if ...
- [最短路]信使(msner)
[题目描述] 战争时期,前线有n个哨所,每个哨所可能会与其他若干个哨所之间有通信联系.信使负责在哨所之间传递信息,当然,这是要花费一定时间的(以天为单位).指挥部设在第一个哨所.当指挥部下达一个命令后 ...
- php将html转为图片
在服务器端解析将编译好的html转换为图片. 由于html一般由客户端浏览器解析,服务器端不能直接解析html代码.所以我们需要借助php类库及扩展完成这一需求. 文件转换过程为 html -> ...
- python学习笔记 python实现k-means聚类
# -*- coding: utf-8 -*- """ Created on Thu Mar 16 14:52:58 2017 @author: Jarvis " ...
- Python函数篇
1.函数名的命名规则: 函数名必须以下划线或字母开头,可以包含任意字母.数字或下划线的组合.不能使用任何的标点符号: 函数名是区分大小写的. 函数名不能是保留字. 2. 形参和实参 形参:形式参数,不 ...
- Redis 高可用集群
Redis 高可用集群 Redis 的集群主从模型是一种高可用的集群架构.本章主要内容有:高可用集群的搭建,Jedis连接集群,新增集群节点,删除集群节点,其他配置补充说明. 高可用集群搭建 集群(c ...
- codeforces 895A Pizza Separation 枚举
codeforces 895A Pizza Separation 题目大意: 分成两大部分,使得这两部分的差值最小(注意是圆形,首尾相连) 思路: 分割出来的部分是连续的,开二倍枚举. 注意不要看成0 ...