五、数据类型5.复杂数据类型1.指针
指针就是内存地址
*赋值:var ptr int = &age
func main(){ var age int = 18 //&符号+变量 就可以获取这个变量内存的地址 fmt.Println(&age) //0xc0000a2058 //定义一个指针变量: //var代表要声明一个变量 //ptr 指针变量的名字 //ptr对应的类型是:*int 是一个指针类型 (可以理解为 指向int类型的指针) //&age就是一个地址,是ptr变量的具体的值 var ptr *int = &age fmt.Println(ptr) fmt.Println("ptr本身这个存储空间的地址为:",&ptr) //想获取ptr这个指针或者这个地址指向的那个数据: fmt.Printf("ptr指向的数值为:%v",*ptr) //ptr指向的数值为:18}
取值:*ptr = 21
总结:最重要的就是两个符号:
1.& 取内存地址
2.* 根据地址取值
指针细节
【1】可以通过指针改变指向值
func main(){ var num int = 10 fmt.Println(num) var ptr *int = &num // 改变值为20 *ptr = 20 fmt.Println(num)}
【2】指针变量接收的一定是地址值
【3】指针变量的地址不可以不匹配
PS:*float32意味着这个指针指向的是float32类型的数据,但是&num对应的是int类型的不可以。
【4】基本数据类型(又叫值类型),都有对应的指针类型,形式为数据类型,
比如int的对应的指针就是int, float32对应的指针类型就是*float32。依次类推。
6.标识符的使用标识符定义规则:
四个注意:不可以以数字开头,严格区分大小写,不能包含空格,不可以使用Go中的保留关键字
起名规则:
(1)包名:尽量保持package的名字和目录保持一致,尽量采取有意义的包名,简短,有意义,和标准库不要冲突
1.为什么之前在定义源文件的时候,一般我们都用package main 包 ?
main包是一个程序的入口包,所以你main函数它所在的包建议定义为main包,如果不定义为main包,那么就不能得到可执行文件。
(2)变量名、函数名、常量名 : 采用驼峰法。
(3)如果变量名、函数名、常量名首字母大写,则可以被其他的包访问;
注意:
import导入语句通常放在文件开头包声明语句的下面。
导入的包名需要使用双引号包裹起来。
包名是从$GOPATH/src/后开始计算的,使用/进行路径分隔。
需要配置一个环境变量:GOPATH
并且需要:go env -w GO111MODULE=off
外部包使用
六、运算符1.算数运算符
算术运算符:+ ,-,*,/,%,++,–
package mainimport "fmt"func main(){ //+加号: //1.正数 2.相加操作 3.字符串拼接 var n1 int = +10 fmt.Println(n1) var n2 int = 4 + 7 fmt.Println(n2) var s1 string = "abc" + "def" fmt.Println(s1) // /除号: fmt.Println(10/3) //两个int类型数据运算,结果一定为整数类型 fmt.Println(10.0/3)//浮点类型参与运算,结果为浮点类型 // % 取模 等价公式: a%b=a-a/b*b fmt.Println(10%3) // 10%3= 10-10/3*3 = 1 fmt.Println(-10%3) fmt.Println(10%-3) fmt.Println(-10%-3) //++自增操作: var a int = 10 a++ fmt.Println(a) a-- fmt.Println(a) //++ 自增 加1操作,--自减,减1操作 //go语言里,++,--操作非常简单,只能单独使用,不能参与到运算中去 //go语言里,++,--只能在变量的后面,不能写在变量的前面 --a ++a 错误写法}
2.赋值运算符
package mainimport "fmt"func main(){ var num1 int = 10 fmt.Println(num1) var num2 int = (10 + 20) % 3 + 3 - 7 //=右侧的值运算清楚后,再赋值给=的左侧 fmt.Println(num2) var num3 int = 10 num3 += 20 //等价num3 = num3 + 20; fmt.Println(num3)}
3.关系运算符
关系运算符:==,!=,>, =,<=
关系运算符的结果都是bool型,也就是要么是true,要么是false
package mainimport "fmt"func main(){ fmt.Println(5==9)//判断左右两侧的值是否相等,相等返回true,不相等返回的是false, ==不是= fmt.Println(5!=9)//判断不等于 fmt.Println(5>9) fmt.Println(5=9) fmt.Println(5<=9)}
4.逻辑运算符
逻辑运算符:&&(逻辑与/短路与),||(逻辑或/短路或),!(逻辑非)
package mainimport "fmt"func main(){ //与逻辑:&& :两个数值/表达式只要有一侧是false,结果一定为false //也叫短路与:只要第一个数值/表达式的结果是false,那么后面的表达式等就不用运算了,直接结果就是false -->提高运算效率 fmt.Println(true&&true) fmt.Println(true&&false) fmt.Println(false&&true) fmt.Println(false&&false) //或逻辑:||:两个数值/表达式只要有一侧是true,结果一定为true //也叫短路或:只要第一个数值/表达式的结果是true,那么后面的表达式等就不用运算了,直接结果就是true -->提高运算效率 fmt.Println(true||true) fmt.Println(true||false) fmt.Println(false||true) fmt.Println(false||false) //非逻辑:取相反的结果: fmt.Println(!true) fmt.Println(!false)}
5.优先级
为了提高优先级,可以加()
6.获取用户终端输入
要传入地址变量,因为scan内部是值改变,只有传入地址变量,才能影响到地址变量的值
package mainimport "fmt"func main(){ //实现功能:键盘录入学生的年龄,姓名,成绩,是否是VIP //方式1:Scanln var age int // fmt.Println("请录入学生的年龄:") //传入age的地址的目的:在Scanln函数中,对地址中的值进行改变的时候,实际外面的age被影响了 //fmt.Scanln(&age)//录入数据的时候,类型一定要匹配,因为底层会自动判定类型的 var name string // fmt.Println("请录入学生的姓名:") // fmt.Scanln(&name) var score float32 // fmt.Println("请录入学生的成绩:") // fmt.Scanln(&score) var isVIP bool // fmt.Println("请录入学生是否为VIP:") // fmt.Scanln(&isVIP) //将上述数据在控制台打印输出: //fmt.Printf("学生的年龄为:%v,姓名为:%v,成绩为:%v,是否为VIP:%v",age,name,score,isVIP) //方式2:Scanf fmt.Println("请录入学生的年龄,姓名,成绩,是否是VIP,使用空格进行分隔") fmt.Scanf("%d %s %f %t",&age,&name,&score,&isVIP) //将上述数据在控制台打印输出: fmt.Printf("学生的年龄为:%v,姓名为:%v,成绩为:%v,是否为VIP:%v",age,name,score,isVIP)}
七、流程控制1.if
if 条件表达式 { 逻辑代码}
当条件表达式为ture时,就会执行得的代码。
条件表达式左右的()可以不写,也建议不写
if和表达式中间,一定要有空格
在Golang中,{}是必须有的,就算你只写一行代码。
if 条件表达式 { 逻辑代码1} else { 逻辑代码2}
if 条件表达式1 { 逻辑代码1} else if 条件表达式2 { 逻辑代码2}.......else { 逻辑代码n}
2.Switch
switch 表达式 { case 值1,值2,.….: 语句块1 case 值3,值4,...: 语句块2 .... default: 语句块}
注意
switch后是一个表达式(即:常量值、变量、一个有返回值的函数等都可以)
case后面的值如果是常量值(字面量),则要求不能重复
case后的各个值的数据类型,必须和 switch 的表达式数据类型一致
case后面可以带多个值,使用逗号间隔。比如 case 值1,值2…
case后面不需要带break
default语句不是必须的,位置也是随意的。
switch后也可以不带表达式,当做if分支来使用
switch后也可以直接声明/定义一个变量,分号结束,不推荐
switch穿透,利用fallthrough关键字,如果在case语句块后增加fallthrough ,则会继续执行下一个case,也叫switch穿透。
3.for循环
for的初始表达式 不能用var定义变量的形式,要用:=
for 初始表达式; 布尔表达式; 迭代因子 { 循环体;}
var sum int = 0 for i := 1 ; i <= 5 ; i++ { sum += i }
for range
(键值循环) for range结构是Go语言特有的一种的迭代结构,在许多情况下都非常有用,for range 可以遍历数组、切片、字符串、map 及通道,for range 语法上类似于其它语言中的 foreach 语句
一般形式为:
for key, val := range coll { ...}
for i , value := range str { fmt.Printf("索引为:%d,具体的值为:%c \n",i,value) } //对str进行遍历,遍历的每个结果的索引值被i接收,每个结果的具体数值被value接收 //遍历对字符进行遍历的
4.关键字1.break
1.switch分支中,每个case分支后都用break结束当前分支,但是在go语言中break可以省略不写。
2.break可以结束正在执行的循环
标签的使用
package mainimport "fmt"func main(){ //双重循环: label2: for i := 1; i <= 5; i++ { for j := 2; j <= 4; j++ { fmt.Printf("i: %v, j: %v \n",i,j) if i == 2 && j == 2 { break label2 //结束指定标签对应的循环 } } } fmt.Println("-----ok")}
注意:如果那个标签没有使用到 的话,那么标签不用加,否则报错:定义未使用
结果:
2.continue
continue的作用结束这一层循环,继续进行下一层
package mainimport "fmt"func main(){ //双重循环: for i := 1; i <= 5; i++ { for j := 2; j <= 4; j++ { if i == 2 && j == 2 { continue } fmt.Printf("i: %v, j: %v \n",i,j) } } fmt.Println("-----ok")}
3.goto
【1】Golang的 goto 语句可以无条件地转移到程序中指定的行。
【2】goto语句通常与条件语句配合使用。可用来实现条件转移.
【3】在Go程序设计中一般不建议使用goto语句,以免造成程序流程的混乱。
【4】代码展示:
package mainimport "fmt"func main(){ fmt.Println("hello golang1") fmt.Println("hello golang2") if 1 == 1 { goto label1 //goto一般配合条件结构一起使用 } fmt.Println("hello golang3") fmt.Println("hello golang4") fmt.Println("hello golang5") fmt.Println("hello golang6") label1: fmt.Println("hello golang7") fmt.Println("hello golang8") fmt.Println("hello golang9")}
4.return
package mainimport "fmt"func main(){ for i := 1; i <= 100; i++ { fmt.Println(i) if i == 14 { return //结束当前的函数 } } fmt.Println("hello golang")}
八、函数1.基本语法
func 函数名(形参列表)(返回值类型列表){ 执行语句.. return + 返回值列表}
//自定义函数:功能:两个数相加:func call01 (num1 int ,num2 int) (int){//如果返回值类型就一个的话,那么()是可以省略不写的return num1 + num2}
2.返回多个
省略返回值:
3.不支持重载
Golang中函数不支持重载
4.可变数量的形参
Golang中支持可变参数 (如果你希望函数带有可变数量的参数)
package mainimport "fmt"//定义一个函数,函数的参数为:可变参数 ... 参数的数量可变//args...int 可以传入任意多个数量的int类型的数据 传入0个,1个,,,,n个func test (args...int){ //函数内部处理可变参数的时候,将可变参数当做切片来处理 //遍历可变参数: for i := 0; i < len(args); i++ { fmt.Println(args[i]) }}func main(){ test() fmt.Println("--------------------") test(3) fmt.Println("--------------------") test(37,58,39,59,47)}
5.修改数值使用地址传递
基本数据类型和数组默认都是值传递的,即进行值拷贝。在函数内修改,不会影响到原来的值。
以值传递方式的数据类型,如果希望在函数内的变量能修改函数外的变量,可以传入变量的地址&,函数内以指针的方式操作变量。从效果来看类似引用传递。
6.函数添加变量名称
在Go中,函数也是一种数据类型,可以赋值给一个变量,则该变量就是一个函数类型的变量了。通过该变量可以对函数调用。
package mainimport "fmt"//定义一个函数:func test(num int){ fmt.Println(num)}func main(){ //函数也是一种数据类型,可以赋值给一个变量 a := test//变量就是一个函数类型的变量 fmt.Printf("a的类型是:%T,test函数的类型是:%T \n",a,test)//a的类型是:func(int),test函数的类型是:func(int) //通过该变量可以对函数调用 a(10) //等价于 test(10)}
7.函数作为形参
函数既然是一种数据类型,因此在Go中,函数可以作为形参,并且调用
(把函数本身当做一种数据类型)
package mainimport "fmt"//定义一个函数:func test(num int){ fmt.Println(num)}//定义一个函数,把另一个函数作为形参:func test02 (num1 int ,num2 float32, testFunc func(int)){ fmt.Println("-----test02")}func main(){ //函数也是一种数据类型,可以赋值给一个变量 a := test//变量就是一个函数类型的变量 fmt.Printf("a的类型是:%T,test函数的类型是:%T \n",a,test)//a的类型是:func(int),test函数的类型是:func(int) //通过该变量可以对函数调用 a(10) //等价于 test(10) //调用test02函数: test02(10,3.19,test) test02(10,3.19,a)}
8.自定义数据类型
为了简化数据类型定义,Go支持自定义数据类型
基本语法: type 自定义数据类型名 数据类型
可以理解为 : 相当于起了一个别名
例如:type mylnt int —–》这时mylnt就等价int来使用了.
支持对函数返回值命名
升级写法:对函数返回值命名,里面顺序就无所谓了,顺序不用对应
9.包的引入
不可能把所有的函数放在同一个源文件中,可以分门别类的把函数放在不同的原文件中
1.简单实例
项目结构
main包
db包
2.包的命名
可以给包取别名,
取别名后,原来的包名就不能使用了
10.init函数
init函数:初始化函数,可以用来进行一些初始化的操作
每一个源文件都可以包含一个init函数,该函数会在main函数执行前,被Go运行框架调用。
11.包执行流程 12.多个包执行流程
13.匿名函数
在定义匿名函数时就直接调用,这种方式匿名函数只能调用一次(用的多)
将匿名函数赋给一个变量(该变量就是函数变量了),再通过该变量来调用匿名函数(用的少)
让一个匿名函数,可以在整个程序中有效呢?将匿名函数给一个全局变量就可以了
package mainimport "fmt"var Func01 = func (num1 int,num2 int) int{ return num1 * num2}func main(){ //定义匿名函数:定义的同时调用 result := func (num1 int,num2 int) int{ return num1 + num2 }(10,20) fmt.Println(result) //将匿名函数赋给一个变量,这个变量实际就是函数类型的变量 //sub等价于匿名函数 sub := func (num1 int,num2 int) int{ return num1 - num2 } //直接调用sub就是调用这个匿名函数了 result01 := sub(30,70) fmt.Println(result01) result02 := sub(30,70) fmt.Println(result02) result03 := Func01(3,4) fmt.Println(result03)}
14.闭包
闭包就是一个函数和与其相关的引用环境组合的一个整体
package mainimport "fmt"//函数功能:求和//函数的名字:getSum 参数为空//getSum函数返回值为一个函数,这个函数的参数是一个int类型的参数,返回值也是int类型func getSum() func (int) int { var sum int = 0 return func (num int) int{ sum = sum + num return sum }}//闭包:返回的匿名函数+匿名函数以外的变量numfunc main(){ f := getSum() fmt.Println(f(1))//1 fmt.Println(f(2))//3 fmt.Println(f(3))//6 fmt.Println(f(4))//10}
匿名函数中引用的那个变量会一直保存在内存中,可以一直使用
1.本质
闭包本质依旧是一个匿名函数,只是这个函数引入外界的变量/参数
匿名函数+引用的变量/参数 = 闭包
2.特点
(1)返回的是一个匿名函数,但是这个匿名函数引用到函数外的变量/参数 ,因此这个匿名函数就和变量/参数形成一个整体,构成闭包。
(2)闭包中使用的变量/参数会一直保存在内存中,所以会一直使用—》意味着闭包不可滥用(对内存消耗大)
15.defer
在函数中,程序员经常需要创建资源,为了在函数执行完毕后,及时的释放资源,Go的设计者提供defer关键字
遇到defer关键字,会将后面的代码语句压入栈中,也会将相关的值同时拷贝入栈中,不会随着函数后面的变化而变化。
应用场景:
比如你想关闭某个使用的资源,在使用的时候直接随手defer,因为defer有延迟执行机制(函数执行完毕再执行defer压入栈的语句),
所以你用完随手写了关闭,比较省心,省事
16.系统函数
【1】统计字符串的长度,按字节进行统计:
len(str)
使用内置函数也不用导包的,直接用就行
【2】字符串遍历:
(1)利用方式1:for-range键值循环:
(2)r:=[]rune(str)
【3】字符串转整数:
n, err := strconv.Atoi("66")
【4】整数转字符串:
str = strconv.Itoa(6887)
【5】查找子串是否在指定的字符串中:
strings.Contains(“javaandgolang”, “go”)
【6】统计一个字符串有几个指定的子串:
strings.Count("javaandgolang","a")
【7】不区分大小写的字符串比较:
strings.EqualFold("go" , "Go")
【8】返回子串在字符串第一次出现的索引值,如果没有返回-1 :
strings.lndex("javaandgolang" , "a")
【9】字符串的替换:
strings.Replace(“goandjavagogo”, “go”, “golang”, n)
n可以指定你希望替换几个,如果n=-1表示全部替换,替换两个n就是2
【10】按照指定的某个字符,为分割标识,将一个学符串拆分成字符串数组:
strings.Split(“go-python-java”, “-“)
【11】将字符串的字母进行大小写的转换:
strings.ToLower(“Go”)// go
strings.ToUpper”go”)//Go
【12】将字符串左右两边的空格去掉:
strings.TrimSpace(” go and java “)
【13】将字符串左右两边指定的字符去掉:
strings.Trim(“golang “, ” ~”)
【14】将字符串左边指定的字符去掉:
strings.TrimLeft(“golang“, “~”)
【15】将字符串右边指定的字符去掉:
strings.TrimRight(“golang“, “~”)
【16】判断字符串是否以指定的字符串开头:
strings.HasPrefix(“http://java.sun.com/jsp/jstl/fmt”, “http”)
【17】判断字符串是否以指定的字符串结束:
strings.HasSuffix(“demo.png”, “.png”)
17.日期和时间
时间和日期的函数,需要到入time包,所以你获取当前时间,就要调用函数Now函数:
package mainimport ( "fmt" "time")func main(){ //时间和日期的函数,需要到入time包,所以你获取当前时间,就要调用函数Now函数: now := time.Now() //Now()返回值是一个结构体,类型是:time.Time fmt.Printf("%v ~~~ 对应的类型为:%T\n",now,now) //2021-02-08 17:47:21.7600788 +0800 CST m=+0.005983901 ~~~ 对应的类型为:time.Time //调用结构体中的方法: fmt.Printf("年:%v \n",now.Year()) fmt.Printf("月:%v \n",now.Month())//月:February fmt.Printf("月:%v \n",int(now.Month()))//月:2 fmt.Printf("日:%v \n",now.Day()) fmt.Printf("时:%v \n",now.Hour()) fmt.Printf("分:%v \n",now.Minute()) fmt.Printf("秒:%v \n",now.Second())}
【2】日期的格式化:
(1)将日期以年月日时分秒按照格式输出为字符串:
//Printf将字符串直接输出: fmt.Printf("当前年月日: %d-%d-%d 时分秒:%d:%d:%d \n",now.Year(),now.Month(), now.Day(),now.Hour(),now.Minute(),now.Second()) //Sprintf可以得到这个字符串,以便后续使用: datestr := fmt.Sprintf("当前年月日: %d-%d-%d 时分秒:%d:%d:%d \n",now.Year(),now.Month(), now.Day(),now.Hour(),now.Minute(),now.Second()) fmt.Println(datestr)
(2)按照指定格式:
(2)按照指定格式:
//这个参数字符串的各个数字必须是固定的,必须这样写 datestr2 := now.Format("2006/01/02 15/04/05") fmt.Println(datestr2) //选择任意的组合都是可以的,根据需求自己选择就可以(自己任意组合)。 datestr3 := now.Format("2006 15:04") fmt.Println(datestr3)
18.内置函数
内置函数存放位置:
在builtin包下,使用内置函数也的,直接用就行
1.常用函数1.len函数:
统计字符串的长度,按字节进行统计
2.new函数
分配内存,主要用来分配值类型(int系列, float系列, bool, string、数组和结构体struct)
3.make函数
分配内存,主要用来分配引用类型(指针、slice切片、map、管道chan、interface 等)