Golang 类型定义总结手册| 面试最基础

Posted by SeaYuns Space on Tuesday, January 29, 2019 | ,阅读约 4 分钟

变量 var

  • 关键字是 var ,定义后须被调用
  • 支持多个同时定义
  • 支持使用 := 缺省定义

变量定义(声明)

//使用var 关键字 进行变量定义 : var + 变量名 + 变量类型
//Var name type
var aa int = 10
//var name 省略类型
var aa = 10
// 多个进行定义,也可省略类型
var Mon, Tues, Wed, Thur, Fri, Sat, Sun int
var (
    Monday    int
    Tuesday   int
    Wednesday int
    Thursday  int
    Friday    int
    Saturday  int
    Sunday    int
)
// 使用 := 快捷定义,但":=" 结构不能在函数外使用
 var aa := 10
 //多个定义
 aa, bb :=10, '10'
 //声明但可不使用
 aa,_ := 10, '10'

变量调用

//变量调用分为 局部变量 和 全局变量
package main

//import
import (
	"fmt"
)

//全局变量定义 aa ss
var (
	aa = 3
	ss = "kkk"
)

//局部变量定义
func variableZeroValue() {
	var a int
	var s string
	//fmt.Println(a, s)
	fmt.Printf("%d %q \n", a, s)
}
func main() {
	//fmt.Println("helloworld")
	variableZeroValue()
	}

常量 const

  • 关键字为 const, 可不被调用
  • 枚举类型 iota
  • 常量不可改变

常量定义

//const name type
//const name
//:=
//多常量
const(
aa = 1
bb =2//使用iota, 从0开始,iota 自增被打断,须显式恢复
const (
		b = 1 << (10 * iota)
		kb
		mb
		gb
	)

常量调用

  • 常量使用与变量相似 ,故不赘述
  • 与c语言不同,常量不能通过指针进行强制更改

数组 arr

  • 在Go语言中数组是固定长度的数据类型,它包含相同类型的连续的元素,这些元素可以是内建类型,像数字和字符串,也可以是结构类型,元素可以通过唯一的索引值访问,从0开始。
  • 使用 var 关键字

数组定义

//定义 长度是5的arr
var arr [5]int
//定义并初始化
var arr [5]int = [5]int{1,2,3,4,5}
//快速定义
arr := [5]int{1,2,3,4,5}
//不知长度
arr := [...]int{1,2,3,4}
//部分初始化, 其他位置初始化为0
arr := [5]int{1:1,3:4}

调用

package main
import "fmt"

func main() {
    array := [2]int{}
    fmt.Printf("数组长度:%d,数组容量:%d\n", len(array), cap(array))
}
//数组长度:2,数组容量:2

切片 Slice

  • “动态数组” ,长度不固定,可以追加 append (name, 5)
  • 通过内部指针和相关属性应用数组片段
  • type, len, cap 为Slice三个属性, 如果slice == nil 则len,cap均为零

定义

//定义一个切片,不用大小
var name []int
//定义并初始化
var aa []int = []int{1,2,3}
var bb = []int{1,2,3}
//通过内置函数make 进行定义 make (type, len, cap)
var aa []int = make([]int, 3)
var aa = make([]int, 3)
var aa = make([]int, 3, 3)
aa := make([]int, 3)
//通过切数组得到 arr := [5]int{1,2,3,4,5}
aa := arr[:len(arr)-1]

调用

切片拷贝,使用copy() 函数 copy 在两个 slice 间复制数据,复制长度以 len 小的为准。两个 slice 可指向同一底层数组,允许元素区间重叠。

package main
import "fmt"
func main() {    
	var a = []int{1, 2, 3, 4, 5}
    b := []int{100, 200}
    copy(a, b)
    fmt.Println(a, b)
    }

运行结果:

[100 200 3 4 5]  [100 200]

容器 Map

  • key- value 表
  • 无序,散列表hash完成
  • 可使用内置函数delete(m, “key”), 不可以在map上使用cap()方法。

定义

//声明
var m map[int]string
//声明并初始化
var m map[int]string {1;"1",2;'2'}
//通过make 创建
var m1 = make(map[int]string, 10)
m2 = make(map[int]string, 10)

调用

//增
m[3] = "3"
//删,删除操作,如果 key 不存在,不会出错
delete(m, "3")
//改
m[1] ='2'
//查
if _, ok = m[3]; ok{
 fmt.Println("find it")
}
//长
len := len(m)

管道 Channel

  • 看成管道, 可单双向使用, chan 定义

  • 无缓冲的与有缓冲channel有着重大差别,那就是一个是同步的 一个是非同步的。 比如

无缓冲chan:ch1:=make(chan int)

有缓冲chan:ch2:=make(chan int,1)

无缓冲: ch1<-1 不仅仅是向 c1 通道放 1,而是一直要等有别的携程 <-ch1 接手了这个参数,那么ch1<-1才会继续下去,要不然就一直阻塞着。

有缓冲: ch2<-1 则不会阻塞,因为缓冲区大小是1(其实是缓冲大小为0),只有当放第二个值的时候,第一个还没被人拿走,这时候才会阻塞。

缓冲区是内部属性,并非类型构成要素。

  • 普通 channel 可以隐式转为只读channel或只写channel,反过来则不行
  • 内置函数 len() 返回未被读取的缓冲元素数量,使用内置函数 cap() 返回缓冲区大小
  • 使用内置函数 close() 进行关闭 chan,向已经关闭的 channel 发送数据会引发 panic 错误。

定义

//使用chan 定义
var aa chan int
var aa chan string
var aa chan map[int]string
//make 定义 
var aa = make(chan int) //无缓冲
var aa = make(chan int, 10) //有缓冲
//只读 
var aa <-chan int
var aa = make(<-chan int, 10)
//只写
var aa chan<- int
var aa = make(chan<-int, 10) 

调用

//开
var aa = make(chan int, 3)
//放
aa <- 1
//关
close(aa)

指针

  • *(取值符) 来获取指针所指向的内容 , &(取地址符)来获取该变量的指针。
  • 不可加减乘除,Go语言是不允许两个指针类型进行转换的
  • 空指针 nil
  • unsafe.Pointer 类型用于表示任意类型的指针

定义

//使用var 进行
var p *int
//使用:= 定义
var str string
p := &str
//指针变量的指针
var **p  int
//time.Time (一个结构体)值的指针 
var t *time.Time

结构体

  • 结构体是由一系列具有相同类型或不同类型的数据构成的数据集合。

  • struct 特点 1、用来自定义复杂数据结构 2、struct里面可以包含一个或多个字段(属性) 3、struct类型可以定义方法,注意和函数的区分 4、struct类型是值类型 5、struct类型可以嵌套 6、结构体是用户单独定义的类型,不能和其他类型进行强制转换 7、Go中的struct没有构造函数,一般可以使用工厂模式来解决这个问题 8、我们可以为struct中的每个字段,写上一个tag。这个tag可以通过反射的机制获取到,最常用的场景就是json序列化和反序列化 9、访问结构体成员, 用 “.” 来连接,格式为:“结构体.成员名”

定义

//全局或函数内定义类型
type bight int64
//定义新类型,不能同时嵌入某一类型和其指针类型(名字相同)
type moreType struct{
	linux string
	win string
	debin string
}
type lin stuct{
	*linux
}
//声明struct结构的时候,在属性的右侧用小米点括起来的内容叫标签(Tag),在转换成其它数据格式的时候,会使用其中特定的字段作为键值。例如转成json格式
	user := &User{UserId: 1, UserName: "helloworld"}
	json_user, _ := json.Marshal(user)
	fmt.Printf("struct User echo : %v\n", string(json_user))

	user_tag := &UserTag{UserId: 1, UserName: "helloworld"}
	json_user_tag, _ := json.Marshal(user_tag)
	fmt.Printf("struct UserTag echo : %v\n", string(json_user_tag))

函数

定义

//无参数
func funName(){
}
//有参数
func funName(x, y int){
}
//一返回值
func funName(x, y int) int{
}
//多返回
fun funNmame(x,y int)(ret int, err error){
}
//只有类型
fun funNmame(intint)(int, error){
}

1、Go语言函数中的参数不支持默认值。

2、无论是值传递,还是引用传递,传递给函数的都是变量的副本,不过,值传递是值的拷贝。引用传递是地址的拷贝,一般来说,地址拷贝更为高效。而值拷贝取决于拷贝的对象大小,对象越大,则性能越低。

3、map、slice、chan、指针、interface默认以引用的方式传递。

4、函数的可变参数只能有一个,且必须是最后一个。

5、在参数赋值时可以不用用一个一个的赋值,可以直接传递一个数组或者切片,特别注意的是在参数后加上“…”即可。

特殊函数

//匿名函数
var sum func(int, int) int = func(x, y int)int{return x+y}
sum := func(x, y int)int{return x+y}

//递归函数
func factorial(i int) int {
    if i <= 1 {
        return 1
    }
    return i * factorial(i-1)
}

延时调用 defer 注意

  • 当defer被声明时,参数被实时解析
  • defer执行顺序为先进后出。
  • defer 可以读取有名返回值