go语言基本操作---三
变量的内存和变量的地址
指针是一个代表着某个内存地址的值。这个内存地址往往是在内存中存储的另一个变量的值的起始位置。Go语言对指针的支持介于java语言和C/C+语言之间,它即没有想Java语言那样取消了代码对指针的直接操作的能力,也避免了C/C+语言中由于对指针的滥用而造成的安全和可靠性问题。
package mainimport "fmt"func main() {var a int = 10//每个变量有2层含义,变量的内存,变量的地址fmt.Printf("a = %d\n", a) //变量的内存 10fmt.Printf("&a = %p\n", &a) //变量的地址 0xc00000a0c8
}
指针变量的基本使用
//保存某个变量的地址,需要指针类型,*int保存int的地址 **int保存*int地址
var a int = 10
var p *int //定义只是特殊的声明
p = &a //指针变量指向谁,就把谁的地址赋值给指针变量
fmt.Printf("p = %p, &a = %p\n", p, &a) //p = 0xc00000a0c8, &a = 0xc00000a0c8*p = 666 //*p操作的不是p的内存是所指向的内存a
fmt.Printf("*p = %d,a = %d\n", *p, a) //*p = 666,a = 666
不要操作没有合法的指向的内存
var p *intp = nilfmt.Println("p = ", p) //<nil>//*p = 666 //err 因为p没有合法的执行var a intp = &a*p = 666fmt.Println("a = ", a)
new函数的使用
表达式new(T)将创建一个T类型的匿名变量,所做的是T类型的新值分配并清零一块内存空间,然后将这块内存空间的地址作为结果返回,而这个结果就是指向这个新的T类型值的指针值,返回的指针类型为*T。
//a :=10 //整型变量avar p *int
//指向一个合法内存
p = new(int) //p是*int,执行int类型 new为没有名字的内存*p = 666
fmt.Println("*p = ", *p)q :=new(int)
*q = 777
我们只需使用new函数,无需担心其内存的生命周期或怎样将其删除,因为go语言的内存管理系统会帮我们打理一切。
普通变量做函数参数(值传递)
package mainimport "fmt"func swap(a, b int) {a, b = b, afmt.Printf("swap: a = %d, b = %d\n", a, b) // 20 10
}func main() {a, b := 10, 20//通过一个函数交换a和b的内存swap(a, b) //变量本身传递,值传递fmt.Printf("main: a = %d, b = %d\n", a, b) //10 20
}
指针做函数参数(地址传递)
package mainimport "fmt"func swap(p1, p2 *int) {*p1, *p2 = *p2, *p1fmt.Printf("swap: *p1 = %d, *p2 = %d\n", *p1, *p2) // 20 10
}func main() {a, b := 10, 20//通过一个函数交换a和b的内存swap(&a, &b) //地址传递fmt.Printf("main: a = %d, b = %d\n", a, b) //20 10
}
数组的基本操作
数组是指一系列同一类型数据的集合。数组中包含的每个数据被称为数组元素,一个数组包含的元素个数被称为数组长度。
package mainimport "fmt"func main() {//定义一个数组[10]int和[5]int是不同类型//[数组],这个数字作为数组元素个数var a [10]intvar b [5]intfmt.Printf("len(a) = %d, len(b) = %d\n", len(a), len(b))// 注意:定义数组是,数组元素个数必须是常量//n := 10//var c [n]int //error non-constant array bound n//操作数组元素,从0开始,到len()-1,不对称元素,这个数字,叫下标//这是下标,可以是变量或常量a[0] = 1i := 1a[i] = 2//赋值,每个元素for i := 0; i < len(a); i++ {a[i] = i + 1}//打印 第一个返回下标,第二个返回元素for i, data := range a {fmt.Printf("a[%d] = %d\n", i, data)}
}
数组初始化
//声明定义同时赋值,叫初始化//1 全部初始化var a [5]int = [5]int{1, 2, 3, 4, 5}fmt.Println("a = ", a) //a = [1 2 3 4 5]b := [5]int{1, 2, 3, 4, 5}fmt.Println("b = ", b) //b = [1 2 3 4 5]//部分初始化,没有初始化的元素,自动赋值为0c := [5]int{1, 2, 3}fmt.Println("c = ", c) //c = [1 2 3 0 0]//指定某个元素初始化d := [5]int{2: 10, 4: 20}fmt.Println("d = ", d) //d = [0 0 10 0 20]
二维数组的介绍
//有多个[]就是多少维//有多少个[]就用多少个循环var a [3][4]intk := 0for i := 0; i < 3; i++ {for j := 0; j < 4; j++ {k++a[i][j] = kfmt.Printf("a[%d][%d] = %d, ", i, j, a[i][j])}fmt.Printf("\n")}fmt.Println("a = ", a) //a = [[1 2 3 4] [5 6 7 8] [9 10 11 12]]//有3个元素,每个元素又是一维数组[4]intb := [3][4]int{{1, 2, 3, 4},{5, 6, 7, 8},{9, 10, 11, 12},}fmt.Println("b = ", b) //b = [[1 2 3 4] [5 6 7 8] [9 10 11 12]]//部分初始化没有初始化的值为0c := [3][4]int{{1, 2, 3},{5, 6, 7, 8},{9, 10},}fmt.Println("c = ", c) //c = [[1 2 3 0] [5 6 7 8] [9 10 0 0]]//部分初始化没有初始化的值为0d := [3][4]int{{1, 2, 3},{5, 6, 7, 8},}fmt.Println("d = ", d) //d = [[1 2 3 0] [5 6 7 8] [0 0 0 0]]//部分初始化没有初始化的值为0e := [3][4]int{0: {1, 2, 3, 4}, 1: {5, 6, 7, 8}}fmt.Println("e = ", e) //e = [[0 0 0 0] [5 6 7 8] [0 0 0 0]]
数组比较和赋值
//支持比较,只支持==或!=,比较每一个元素都一样,2个数组比较,数组类型要一样a := [5]int{1, 2, 3, 4, 5}b := [5]int{1, 2, 3, 4, 5}c := [5]int{1, 2, 3}fmt.Println("a == b ", a == b) //truefmt.Println("a == c ", a == c) //false//同类型的数组可以赋值var d [5]intd = afmt.Println("d = ", d) //d = [1 2 3 4 5]
随机数的使用
package mainimport ("fmt""math/rand""time"
)func main() {//设置种子,只需要一次//如果种子参数一样,每次运行程序产生的随机数都一样//rand.Seed(666)//for i := 0; i < 5; i++ {// //产生随机数// fmt.Println("rand = ", rand.Int())//随机数很大的范围//}//rand.Seed(time.Now().UnixNano()) //以当前系统时间作为种子参数//for i := 0; i < 5; i++ {// //产生随机数// fmt.Println("rand = ", rand.Int()) //随机数很大的范围//}rand.Seed(time.Now().UnixNano()) //以当前系统时间作为种子参数for i := 0; i < 5; i++ {//产生随机数fmt.Println("rand = ", rand.Intn(100)) //限制在100内的数}
}
冒泡排序代码实现
rand.Seed(time.Now().UnixNano()) //以当前系统时间作为种子参数var a [10]intfor i := 0; i < len(a); i++ {a[i] = rand.Intn(100) //100以内的随机数fmt.Printf("%d, ", a[i])}fmt.Printf("\n")//冒泡排序 挨着的2个元素比较,升序(大于则交换)for i := 0; i < len(a)-1; i++ {for j := 0; j < len(a)-1-i; j++ {if a[j] > a[j+1] {a[j], a[j+1] = a[j+1], a[j]}}}fmt.Printf("\n排序后:\n")for i := 0; i < len(a); i++ {fmt.Printf("%d, ", a[i])}fmt.Printf("\n")
数组做函数参数是值拷贝
package mainimport "fmt"// 数组做函数参数,他是值传递
// 实参数组的每个元素给形参数组拷贝一份
// 形参数组是实参的复制品,数组越大,拷贝的效率越低
func modify(a [5]int) {a[0] = 666fmt.Println("modify a = ", a)//modify a = [666 2 3 4 5]
}func main() {a := [5]int{1, 2, 3, 4, 5}modify(a) //数组传递过去fmt.Println("main: a = ", a) //main: a = [1 2 3 4 5]
}
数组指针做函数参数
package mainimport "fmt"// P指向实参数组a,他是数组指针
// *p代表指针所指向的内存,就是实参a
func modify(p *[5]int) {(*p)[0] = 666fmt.Println("modify a = ", *p)
}func main() {a := [5]int{1, 2, 3, 4, 5}//modify a = [666 2 3 4 5]//main: a = [666 2 3 4 5]modify(&a) //地址传递fmt.Println("main: a = ", a)
}
切片
切片并不是数组或数组指针,它通过内部指针和相关属性引用数组片段,以实现边长方案。
slice并不是真正意义上的动态数组,而是一个引用类型。slice总是指向一个底层array,slice的声明也可以像array一样,只是不需要长度
a := []int{1, 2, 3, 4, 5}s := a[0:3:5] //左闭右开fmt.Println("s = ", s) // s = [1,2,3]fmt.Println("len(s) = ", len(s)) //长度 //3 3-0fmt.Println("cap(s) = ", cap(s)) //容量 //5 5-0s1 := a[1:4:5] //左闭右开fmt.Println("s = ", s1) // 从下标1开始,取4-1=3个 s = [2 3 4]fmt.Println("len(s) = ", len(s1)) //长度 //3 4-1fmt.Println("cap(s) = ", cap(s1)) //容量 //4 5-1
切片和数组区别
//切片和数组的区别
//数组[]里面的长度是固定的一个常量,数组不能修改长度,len和cap永远都是5
a := [5]int{}
fmt.Printf("len = %d, cap = %d\n", len(a), cap(a)) //len = 5, cap = 5//切片,[]里面为空,或者为....切片的长度或容易可以不固定
s := []int{}
fmt.Printf("1: len = %d, cap = %d\n", len(s), cap(s)) //1: len = 0, cap = 0s = append(s, 11) //给切片末尾追加一个成员
fmt.Printf("append: len = %d, cap = %d\n", len(s), cap(s)) //append: len = 1, cap = 1
切片的创建
//自动推导类型 同时初始化
s1 := []int{1, 2, 3, 4}
fmt.Println("s1 = ", s1) //s1 = [1 2 3 4]//借助make函数 格式make(切片类型,长度,容量)
s2 := make([]int, 5, 10)
fmt.Printf("len = %d, cap = %d\n", len(s2), cap(s2)) //len = 5, cap = 10//没有指定容量,容量和长度一样
s3 := make([]int, 5)
fmt.Printf("len = %d, cap = %d\n", len(s3), cap(s3)) //len = 5, cap = 5
切片截取
array := []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}//[low:high:max] 取下标从low开始的元素,len = high-low,cap=max-lows1 := array[:] //[0:len(array):len(array)] 不指定容量和长度一样fmt.Println("s1 = ", s1) //s1 = [0 1 2 3 4 5 6 7 8 9]fmt.Printf("len = %d, cap = %d\n", len(s1), cap(s1)) //10 10//操作某个元素和数组操作方式一样data := array[0]fmt.Println("data = ", data) //0s2 := array[3:6:7] //a[3] a[4] a[5] len = 3 cap =4fmt.Println("s2 = ", s2) //3 4 5fmt.Printf("len = %d, cap = %d\n", len(s2), cap(s2)) //3 4s3 := array[:6] //从0开始,取6个元素,容量是10 容量为数组长度fmt.Println("s3 = ", s3) //[0 1 2 3 4 5]fmt.Printf("len = %d, cap = %d\n", len(s3), cap(s3)) // 6 10s4 := array[3:] //从下标为3开始,到结尾fmt.Println("s4 = ", s4) //[3 4 5 6 7 8 9]fmt.Printf("len = %d, cap = %d\n", len(s4), cap(s4)) //len = 7, cap = 7
切片和底层数组关系
a := []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}s1 := a[2:5] //从a[2]开始取三个元素
fmt.Println("s1 =", s1) //[2 3 4]
s1[1] = 666
fmt.Println("s1 =", s1) //s1 = [2 666 4]
fmt.Println("a = ", a) //a = [0 1 2 666 4 5 6 7 8 9]s2 := s1[1:7]
fmt.Println("s2 = ", s2) //s2 = [666,4 5 6 7 8]
s2[2] = 777
fmt.Println("s2 = ", s2) //s2 = [666 4 777 6 7 8]
fmt.Println("a = ", a) //a = [0 1 2 666 4 777 6 7 8 9]
内建函数------append
append函数向切片尾部添加数据,返回新的切片对象,容量不够,会自动扩容
s1 := []int{}
fmt.Printf("len = %d ,cap = %d\n", len(s1), cap(s1)) // 0 0
fmt.Println("s1 = ", s1) //[]
//在原切片的末尾添加元素
s1 = append(s1, 1)
s1 = append(s1, 2)
s1 = append(s1, 3)
fmt.Printf("len = %d ,cap = %d\n", len(s1), cap(s1)) // 3 4 容量以两倍的速度增加的
fmt.Println("s1 = ", s1) // 1 2 3s2 := []int{1, 2, 3}
fmt.Println("s2 = ", s2) //1 2 3
s2 = append(s2, 5)
s2 = append(s2, 5)
s2 = append(s2, 5)
fmt.Println("s2 = ", s2) //s2 = [1 2 3 5 5 5]
append扩容特点
一旦超过原底层数组容量,通常以2倍容量重新分配底层数组,并复制原来的数据
//如果超过原来的容量,通常以2倍容量扩容
s := make([]int, 0, 1) //cap 1 len 0
oldCap := cap(s)for i := 0; i < 8; i++ {s = append(s, i)if newCap := cap(s); oldCap < newCap {fmt.Printf("cap: %d ====>%d\n", oldCap, newCap) //cap: 1 ====>2 cap: 2 ====>4 cap: 4 ====>8oldCap = newCap}
}
内建函数copy
srcSlice := []int{1, 2}
dstSlice := []int{6, 6, 6, 6, 6}
copy(dstSlice, srcSlice)
fmt.Println("dst", dstSlice) //dst [1 2 6 6 6]
切片做函数参数
切片为引用传递
package mainimport ("fmt""math/rand""time"
)func InitData(s []int) {rand.Seed(time.Now().UnixNano()) //设置种子for i := 0; i < len(s); i++ {s[i] = rand.Intn(100) //100以内的随机数}
}func BubbleSort(s []int) {n := len(s)for i := 0; i < n-1; i++ {for j := 0; j < n-1-i; j++ {if s[j] > s[j+1] {s[j], s[j+1] = s[j+1], s[j]}}}
}func main() {var n int = 10//创建一个切片,len为ns := make([]int, n)InitData(s)fmt.Println("排序前 s = ", s)BubbleSort(s) //冒泡排序fmt.Println("排序后 s = ", s)
}
猜数字游戏
package mainimport ("fmt""math/rand""time"
)// 创建一个数字
func CreatNum(p *int) {rand.Seed(time.Now().UnixNano())var num intfor {num = rand.Intn(10000)if num >= 1000 {break}}//fmt.Println("num = ", num)*p = num
}// 取出每一位
func GetNum(s []int, num int) {s[0] = num / 1000 //取千位s[1] = num % 1000 / 100 //取百位s[2] = num % 100 / 10 //取十位s[3] = num % 10 //取个位
}// 猜数字逻辑
func OnGame(randSlice []int) {var num intkeySlice := make([]int, 4)for {for {fmt.Printf("请输入一个4位数")fmt.Scan(&num)if 999 < num && num < 10000 {break}fmt.Println("请输入的数不符合要求")}//fmt.Println("num = ", num)GetNum(keySlice, num)//fmt.Println("keySlice = ", keySlice)n := 0for i := 0; i < 4; i++ {if randSlice[i] < keySlice[i] {fmt.Printf("第%d位大了一点\n", i+1)} else if keySlice[i] < randSlice[i] {fmt.Printf("第%d位小了一点\n", i+1)} else {fmt.Printf("第%d位猜对了\n", i+1)n++}}if n == 4 { //4位都猜对了fmt.Println("全部猜对")return}}}func main() {var randNum int //产生随机的数//产生一个4位的随机数CreatNum(&randNum)//fmt.Println("randNum: ", randNum)//切片randSlice := make([]int, 4)//保存这个4位数的每一位GetNum(randSlice, randNum)//fmt.Println("randSlice = ", randSlice)//n1 := 1234 / 1000 //取商//n2 := (1234 % 1000)/100 //取余 结果为234//fmt.Println("n1 = ", n1)//fmt.Println("n2 = ", n2)OnGame(randSlice) //游戏
}
map介绍
Go语言中的map(映射,字典)是一种内置的数据结构,他是一个无序的key-value对的集合。
在一个map里所有的键都是唯一的,而且必须是支持==和!=操作符的类型,切片。函数以及包含切片的结构类型这些类型由于具有引用语句,不能作为映射的键,使用这些类型会造成便于错误
map值可以是任意类型,没有限制。map里所有键的数据类型必须是相同的,值也必须如何,但键值的数据类型可以不相同。
注意:map是无序的,我们无法决定它的返回顺序,所以,每次打印结果的顺序有可能不同。
map的基本操作
//map创建
//定义一个变量,类型为map[int]stringvar m1 map[int]string
fmt.Println("m1 = ", m1) //m1 = map[]//对于map只有len,没有cap
fmt.Println("len = ", len(m1)) //0//可以通过make创建
m2 := make(map[int]string)
fmt.Println("m2 = ", m2) //m2 = map[]
fmt.Println("len = ", len(m2)) //0//可以通过make创建,可以指定长度,只是指定了容量,但是里面却是一个数据也没有
m3 := make(map[int]string, 10)
m3[1] = "mike"
m3[3] = "c++"
m3[2] = "go"
fmt.Println("m3 = ", m3) //m3 = map[1:mike 2:go 3:c++]
fmt.Println("len = ", len(m3)) //3//键值是唯一的
m4 := map[int]string{1: "make",2: "go",3: "c++",//4:"xxxx",
}
fmt.Println("m4 = ", m4) //m4 = map[1:make 2:go 3:c++]
map赋值
//键值是唯一的
m1 := map[int]string{1: "make",2: "go",3: "c++",//4:"xxxx",
}
//赋值,如果已经存在的key值就是修改内容
fmt.Println("m1 = ", m1) //m1 = map[1:make 2:go 3:c++]m1[1] = "python"
fmt.Println("m1 = ", m1) //m1 = map[1:python 2:go 3:c++]
m1[4] = "java" //追加,nap底层自动扩容,和append类似
fmt.Println("m1 = ", m1) //m1 = map[1:python 2:go 3:c++ 4:java]
map遍历
//键值是唯一的
m1 := map[int]string{1: "make",2: "go",3: "c++",//4:"xxxx",
}
//1=========>make
//2=========>go
//3=========>c++
//第一个返回值为key,第二个返回值为value,遍历结果是无序的
for key, value := range m1 {fmt.Printf("%d=========>%s\n", key, value)
}//如何判断一个key值是否存在
//第一个返回值为key所对应的value,第二个返回值为key是否存在的条件,存在ok为true
value, ok := m1[1]
if ok == true {fmt.Println("m[1]= ", value) //m[1] make
} else {fmt.Println("key不存在")
}
map删除
//键值是唯一的
m1 := map[int]string{1: "make",2: "go",3: "c++",//4:"xxxx",
}fmt.Println("m1 = ", m1) //m1 = map[1:make 2:go 3:c++]
delete(m1, 1) //删除key为1的内容
fmt.Println("m1 = ", m1) //m1 = map[2:go 3:c++]
map做函数参数
引用传递
package mainimport "fmt"func test1(m map[int]string) {delete(m, 1)
}func main() {//键值是唯一的m1 := map[int]string{1: "make",2: "go",3: "c++",//4:"xxxx",}fmt.Println("m1 = ", m1) //m1 = map[1:make 2:go 3:c++]test1(m1)fmt.Println("m1 = ", m1) //m1 = map[2:go 3:c++]
}
结构体普通变量初始化
结构体类型:
有时我们需要将不同的类型的数据组合成一个有机的整体,如:一个学生有学号/姓名/性别等属性,显然单独定义以上变量比较繁琐,数据不便于管理。
结构体是一种聚合的数据类型,它是由一系列具有相同类型或不同类型的数据构成的数据集合。每个数据称为结构体的成员。
结构体初始化
package mainimport "fmt"//定义一个结构体类型type Student struct {id intname stringsex byte //字符类型age intaddr string
}func main() {//顺序初始化,每个陈冠必须初始化var s1 Student = Student{1, "mike", 'm', 18, "bj"} //s1 = {1 mike 109 18 bj}fmt.Println("s1 = ", s1)//指定成员初始化,没有初始化的成员自动赋值为零s2 := Student{name: "mike", addr: "bj"}fmt.Println("s2 = ", s2) //s2 = {0 mike 0 0 bj}
}
结构体指针变量初始化
package mainimport "fmt"//定义一个结构体类型type Student struct {id intname stringsex byte //字符类型age intaddr string
}func main() {//顺序初始化,每个陈冠必须初始化 别忘了&var p1 *Student = &Student{1, "mike", 'm', 18, "bj"}fmt.Println("*p1 = ", *p1) //*p1 = {1 mike 109 18 bj}fmt.Println("p1 = ", p1) //p1 = &{1 mike 109 18 bj}//指定成员初始化,没有初始化的成员自动赋值为零p2 := &Student{name: "mike", addr: "bj"}fmt.Println("*p2 = ", *p2) //*p2 = {0 mike 0 0 bj}fmt.Println("p2 = ", p2) //p2 = &{0 mike 0 0 bj}fmt.Printf("p2 type is %T\n ", p2) //p2 type is *main.Student
}
结构体成员的使用:普通变量
package mainimport "fmt"//定义一个结构体类型type Student struct {id intname stringsex byte //字符类型age intaddr string
}func main() {//定义一个结构体普通变量var s Student//操作成员需要使用.运算符s.id = 1s.name = "mike"s.sex = 'm' //字符s.age = 18s.addr = "wh"fmt.Println("s = ", s) //s = {1 mike 109 18 wh}
}
结构体成员的使用:指针变量
package mainimport "fmt"//定义一个结构体类型type Student struct {id intname stringsex byte //字符类型age intaddr string
}func main() {//1 指针有合法指向后,才操作成员// 先定义一个普通结构体变量var s Student//在定义一个指针变量,保存s的地址var p1 *Studentp1 = &s//通过指针操作成员p1.id (*p1).id完全等价,只能使用.运算符p1.id = 1(*p1).name = "mike"p1.sex = 'm'p1.age = 18p1.addr = "bj"fmt.Println("p1 = ", p1) //p1 = &{1 mike 109 18 bj}fmt.Println("*p1 = ", *p1) // *p1 = {1 mike 109 18 bj}//2 通过new生气一个结构体p2 := new(Student)p2.id = 1(*p2).name = "mike"p2.sex = 'm'p2.age = 18p2.addr = "bj"fmt.Println("p2 = ", p2) //p2 = &{1 mike 109 18 bj}fmt.Println("*p2 = ", *p2) // *p2 = {1 mike 109 18 bj}
}
结构体比较和赋值
如果结构体的全部成员都是可以比较的,那么结构体也是可以可以比较的,那样的话两个结构体将可以使用==或!=运算符进行比较,但不支持>或<
package mainimport "fmt"//定义一个结构体类型type Student struct {id intname stringsex byte //字符类型age intaddr string
}func main() {s1 := Student{1, "mike", 'm', 18, "bj"}s2 := Student{1, "mike", 'm', 18, "bj"}s3 := Student{2, "mike", 'm', 18, "bj"}fmt.Println("s1 == s3", s1 == s3) //falsefmt.Println("s1 == s2", s1 == s2) //true//同类型的2个结构体变量可以相互赋值var tmp Studenttmp = s3fmt.Println("tmp = ", tmp) //tmp = {2 mike 109 18 bj}
}
结构体做函数参数:值传递
package mainimport "fmt"//定义一个结构体类型type Student struct {id intname stringsex byte //字符类型age intaddr string
}func test01(student Student) {student.id = 2fmt.Println("test01 student = ", student) //test01 student = {2 mike 109 18 bj}
}
func main() {s := Student{1, "mike", 'm', 18, "bj"}test01(s) //值传递,形参无法改实参fmt.Println("main student = ", s) //main student = {1 mike 109 18 bj}
}
结构体做函数参数:地址传递
package mainimport "fmt"//定义一个结构体类型type Student struct {id intname stringsex byte //字符类型age intaddr string
}func test01(student *Student) {student.id = 2fmt.Println("test01 student = ", *student) //test01 student = {2 mike 109 18 bj}
}
func main() {s := Student{1, "mike", 'm', 18, "bj"}test01(&s) //地址传递(引用传递),形参可以改实参fmt.Println("main student = ", s) //main student = {2 mike 109 18 bj}
}
go语言可见性规则验证
Go语言对关键字的增加非常吝啬,其中没有private,protected,public这样的关键字.
要使某个符号对其他包(package)可见(即可以访问),需要将符号定义为以大写字母开头.
1):如果使用别的包的函数,结构体类型,结构体成员,函数名,类型名,结构体成员变量名,首字母必须大写,可见。
2):如果首字母是小写,只能在同一个包里面使用。
相关文章:
![](https://img-blog.csdnimg.cn/2d219e739abc4f48961a3163f3224e0d.png)
go语言基本操作---三
变量的内存和变量的地址 指针是一个代表着某个内存地址的值。这个内存地址往往是在内存中存储的另一个变量的值的起始位置。Go语言对指针的支持介于java语言和C/C语言之间,它即没有想Java语言那样取消了代码对指针的直接操作的能力,也避免了C/C语言中由…...
![](https://img-blog.csdnimg.cn/4ebfdb66e309443e8ac2332963945071.png#pic_center)
ArcGIS Enterprise + ArcGIS Pro 常用服务类型发布
发布前设置 门户连接 首先Pro需要先连接portal 添加portal门户地址,注意只到WA一级地址,并登录: 登录完成后,右键,设置为活动门户: 1. 发布动态地图服务 关联数据文件夹: 拖拽数据到地图…...
![](https://img-blog.csdnimg.cn/img_convert/bac19964fa206f568e49d8ef2bb8210f.webp?x-oss-process=image/format,png)
优思学院|亲和图案例:寻找六西格玛的项目
什么是亲和图? 亲和图(Affinity Diagram)主要功能在於分类归纳,协助在一堆杂乱无章的资料之中,有系统的归纳出几个大类,以利后续作业。通常先利用头脑风暴(Brainstorming)方式得到大…...
![](https://www.ngui.cc/images/no-images.jpg)
tomcat 的缓存机制
HTTP缓存:Tomcat支持HTTP缓存机制,可以通过设置响应头中的Cache-Control、Expires和ETag等字段来控制缓存策略。这些字段告诉浏览器是否可以缓存响应以及缓存的有效期等信息。 Servlet缓存:Tomcat还提供了Servlet缓存机制,它可以…...
![](https://www.ngui.cc/images/no-images.jpg)
laravel 压缩文件与解压文件
一、引入第三方类 composer require chumper/zipper二、第三方类配置 providers>[Chumper\Zipper\ZipperServiceProvider::class ]aliases > [Zipper > Chumper\Zipper\Zipper::class ]三、压缩解压缩实例 <?php namespace App\Http\Controllers\Upload; use A…...
![](https://www.ngui.cc/images/no-images.jpg)
kind搭建k8s集群用于测试
安装kind 需要先安装go kind基于go开发 #第一种安装方式#修改go源加快下载速度 go env -w GOPROXYhttps://goproxy.cn,direct #直接下载安装kind最新版本 go install sigs.k8s.io/kindlatest #进入GOPATH目录找到bin目录下kind执行程序 移动到环境变量里 mv ./kind /usr/local…...
![](https://img-blog.csdnimg.cn/aa8b756f1d74453a8d1f66d80e86a60b.png)
软件测试人需要掌握的测试知识架构体系(上)
软件计划与可行性研究(问题定义、可行性研究);需求分析;软件设计(概要设计、详细设计);编码;软件测试;运行与维护。 一、软件的生命周期(SDLC) 1、生存周期划分 各阶段…...
![](https://img-blog.csdnimg.cn/79c47ea4f43e450cae8f5a5a84e08180.png)
QT数据库,实现数据库增删改查
QT关于数据库的相关概念 QT将数据库分为三个层次: 数据库驱动层:QSqlDriver、QSqlDriverCreator、QSqlDriverCreatorBase、QSqlDriverPlugin sql接口层:QSqlDatabase、QSqlQuery、QSqlRecord、QSqlError 用户接口层:提供一些模…...
![](https://img-blog.csdnimg.cn/38af7309656640e889e2d683a87808bb.png)
SQL-子查询
SQL 子查询 是指将一个SELECT查询(子查询)的结果用括号括起来作为另一个SQL语句的数据来源或者判断条件...
![](https://img-blog.csdnimg.cn/abeb073ff88541449a723d44dd2dc98f.png)
【8章】Spark编程基础(Python版)
课程资源:(林子雨)Spark编程基础(Python版)_哔哩哔哩_bilibili 第8章 Spark MLlib(6节) 机器学习算法库 (一)MLlib简介 1、机器学习 机器学习可以看做是一门人工智能的科学,该领…...
![](https://img-blog.csdnimg.cn/img_convert/a8af01ea87c6a6accd54ed0b24004d12.png)
桌面应用小程序,一种创新的跨端开发方案
Qt Group在提及2023年有桌面端应用程序开发热门趋势时,曾经提及三点: 关注用户体验:无论您是为桌面端、移动端,还是为两者一起开发应用程序,有一点是可以确定的:随着市场竞争日益激烈,对产品的期…...
![](https://img-blog.csdnimg.cn/7c9ada2421d948a68369053d60e1a193.png)
将本地jar打包到本地maven仓库或maven私服仓库中
将本地jar包打包到本地的maven仓库中的命令: mvn install:install-file -DgroupIdtebie.applib.api -DartifactIdapiclient -Dversion1.0-SNAPSHOT -Dfile本地jar路径 -Dpackagingjar说明: DgroupId pom中的<groupId></groupId> Dartifact…...
![](https://www.ngui.cc/images/no-images.jpg)
java 实现建造者模式
建造者模式(Builder Pattern)是一种创建型设计模式,用于创建一个复杂对象,将对象的构建过程与其表示分离,以便可以使用相同的构建过程来创建不同的表示。在Java中,可以使用建造者模式来构建具有多个属性的对…...
![](https://img-blog.csdnimg.cn/bff2de5f2f214a4fab4052f68d804263.png)
串行FIR滤波器
串行 FIR 滤波器设计 串行设计,就是在 16 个时钟周期内对 16 个延时数据分时依次进行乘法、加法运算,然后在时钟驱动下输出滤波值。考虑到 FIR 滤波器系数的对称性,计算一个滤波输出值的周期可以减少到 8 个。串行设计时每个周期只进行一次乘…...
![](https://img-blog.csdnimg.cn/6f2a430bf1004944a51ae4203b5fd577.png)
Spring Boot 整合 Shiro(后端)
1 Shiro 什么是 Shiro 官网: http://shiro.apache.org/ 是一款主流的 Java 安全框架,不依赖任何容器,可以运行在 Java SE 和 Java EE 项目中,它的主要作用是对访问系统的用户进行身份认证、 授权、会话管理、加密等操作。 …...
![](https://img-blog.csdnimg.cn/4a0a61143f56446caaec0ec2f1be4067.gif)
面试中的自我介绍:首印象决定一切
🌷🍁 博主猫头虎(🐅🐾)带您 Go to New World✨🍁 🦄 博客首页——🐅🐾猫头虎的博客🎐 🐳 《面试题大全专栏》 🦕 文章图文…...
![](https://img-blog.csdnimg.cn/6c0cfd46dcda46e3918c98381d36e3af.png)
深入理解联邦学习——联邦学习的价值
分类目录:《深入理解联邦学习》总目录 毫无疑问,如今我们正经历互联网第四次信息革命,坐拥海量的信息与数据。这些数据如果能够用AI的方式进行解读,将会为人类日常生活带来颠覆性变革。联邦学习作为未来AI发展的底层技术ÿ…...
![](https://img-blog.csdnimg.cn/97bbb0af6183468091eed843d07c3cf2.png)
linux 内存一致性
linux 出现内存一致性的场景 1、编译器优化 ,代码上下没有关联的时候,因为编译优化,会有执行执行顺序不一致的问题(多核单核都会出现) 2、多核cpu乱序执行,cpu的乱序执行导致内存不一致(多核出…...
![](https://www.ngui.cc/images/no-images.jpg)
Vue 如何监听 localstorage的变化
需求 分析 1. 初始想法 computed: {lonlat(){console.log(localStorage.getItem(lonlat))return localStorage.getItem(lonlat)}},watch: {lonlat(newVal,oldVal){console.log(1002,newVal,oldVal)}},我们想着用 计算属性 computed 和 watch 监听实现,但根本没有…...
![](https://www.ngui.cc/images/no-images.jpg)
ActiveMQ使用JDBC持久化消息
为了避免服务器宕机而导致消息丢失,ActiveMQ提供消息持久化机制。 ActiveMQ提供多种消息持久化的方式,如LevelDB Store、KahaDB 、AMQ、JDBC等,详情可以访问官网。 ActiveMQ默认是使用KahaDB持久化消息。在/conf/activemq.xml如下配置&…...
![](https://img-blog.csdnimg.cn/61cf6558c8f840ce99319430eef8795e.png#pic_center)
光环云出席Enjoy出海AIGC主题研讨会,助力企业迎接AI时代机遇与挑战
AIGC的崛起,为2023年的全球化突围之路拓展了想象空间。 从年初至今,OpenAI和ChatGPT高举高打,很大程度上起到了教育市场的作用;此外,Meta推出大模型,Snapchat、Soul、字节等大厂或上线或内测聊天机器人&…...
![](https://img-blog.csdnimg.cn/9a106b5c5e204437a27a20e05e942368.png)
动态规划:路径和子数组问题(C++)
动态规划:路径和子数组问题 路径问题1.不同路径(中等)2.不同路径II(中等)3.下降路径最⼩和(中等)4.地下城游戏(困难) 子数组问题1.最大子数组和(中等…...
![](https://img-blog.csdnimg.cn/89b6f009cee34a2e873cd3f55f1a0f42.png)
微服务-gateway跨域配置
文章目录 一、前言二、gateway跨域配置1、问题描述1.1、什么是跨域请求?1.1.1、同源策略1.1.2. 安全性考虑1.1.3. 跨域攻击 1.2、问题产生原因 2、解决方法2.1、修改配置文件2.2、配置类统一配置2.3、全局跨域拦截器 三、总结 一、前言 在SpringCloud项目中&#x…...
![](https://img-blog.csdnimg.cn/ae5f45bd151e485e886e5db29c53c238.png)
爬虫项目(二):中国大学排名
《Python网络爬虫入门到实战》京东购买地址,这里讲解了大量的基础知识和实战,由本人编著:https://item.jd.com/14049708.html配套代码仓库地址:https://github.com/sfvsfv/Crawer文章目录 分析第一步:获取源码分析第一…...
![](https://img-blog.csdnimg.cn/205ed3aeeeba40f788b86cb7f3c889ea.png)
十二、MySQL(DQL)分组/排序/分页查询如何实现?
总括 select 字段列表 from 表名 [where 条件] (group by)/(order by)/(limit) 分组字段名 分组查询 1、分组查询 (1)基础语法: select 字段列表 from 表名 [where 条件] group by 分组字段名 [having 分组之后的过滤条件] (…...
![](https://www.ngui.cc/images/no-images.jpg)
设计模式概念学习
创建类型 单例模式 饿汉 构建时就创建 懒汉 单线程-访问到的时候才创建多线程-低效率 做法:加锁->若未创建则创建->获取资源->解锁 缺点:效率低,每次访问之前都要加锁,资源创建之后不能被同时被多个线程访问多线程-…...
![](https://img-blog.csdnimg.cn/377c39bf46ca45ee999cdc8efd09f8ae.png)
Spring MVC 五 - DispatcherServlet初始化过程(续)
今天的内容是SpringMVC的初始化过程,其实也就是DispatcherServilet的初始化过程。 Special Bean Types DispatcherServlet委托如下一些特殊的bean来处理请求、并渲染正确的返回。这些特殊的bean是Spring MVC框架管理的bean、按照Spring框架的约定处理相关请求&…...
![](https://www.ngui.cc/images/no-images.jpg)
day36:网编day3,TCP、UDP模型
下载: #include <myhead.h>#define ERR(s) do\ {\fprintf(stderr,"__%d__",__LINE__);\perror(s);\ }while(0) #define PORT 69 #define IP "192.168.115.184"int do_download(int cfd,struct sockaddr_in sin); //int do_upload(); int…...
![](https://img-blog.csdnimg.cn/7c9037d5fc504885b5091a369e2d123c.png)
MySQL——MySQL的基础操作部分
使用命令行登录 mysql -u root -p 直接敲击回车后输入密码即可: 当看到出现“mysql>“的符号之后,就表示已经进入到了MySQL系统中,就可以输入My…...
![](https://img-blog.csdnimg.cn/2aab54be754a4f4580898c955f136560.png#pic_center)
编译OpenWrt内核驱动
编译OpenWrt内核驱动可以参考OpenWrt内部其它驱动的编写例程,来修改成自己需要的驱动 一、OpenWrt源代码获取与编译 1.1、搭建环境 下载OpenWrt的官方源码: git clone https://github.com/openwrt/openwrt.git1.2、安装编译依赖项 sudo apt update -…...
![](https://img-blog.csdnimg.cn/20190805114328295.png)
seo服务指什么意思/太原百度快速优化排名
一、C 变量类型 变量其实只不过是程序可操作的存储区的名称。C 中每个变量都有指定的类型,类型决定了变量存储的大小和布局,该范围内的值都可以存储在内存中,运算符可应用于变量上。 变量的名称可以由字母、数字和下划线字符组成。它必须以…...
![](/images/no-images.jpg)
旅游网站建设与翻译/google搜索引擎
UNIX世界的软件开发大多都是协作式的,因此,Patch(补丁)是一个相当重要的东西,因为几乎所有的大型UNIX项目的普通贡献者,都是通过 Patch来提交代码的。作为最重要的开源项目之一,Linux࿰…...
![](https://img-blog.csdnimg.cn/img_convert/803978759418134cfaa90d7bd591d17e.png)
wordpress特色图像地址/网店推广渠道有哪些
这几天在寒潮的影响下,全国多个地区温度骤降,已经好多年没有感受到这样严酷的冬季了,暖气都撑不住。零下十多度的天气不仅影响人的生活,现在电脑也中招了,确切说是散热——有网友发了一个电脑水冷被冻住的视频…...
![](/images/no-images.jpg)
动漫制作技术就业前景/seo服务套餐
转自:https://www.pinlue.com/article/2020/01/1616/439904883431.html...
![](https://img-blog.csdnimg.cn/20210901155444662.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA5L2O5ZCffua1heWUsQ==,size_18,color_FFFFFF,t_70,g_se,x_16)
建站系统下载 discuz/网络代运营推广
$x array(1,2,3,4,5); var_dump($x);//除去下标在为3底下的值 unset($x[3]); var_dump($x);//除去下标为3的值后面重新排序 $x array_values($x); var_dump($x);...
![](https://img-blog.csdnimg.cn/img_convert/0818b9ca8b590ca3270a3433284dd417.png)
qq云端服务器/seo推广优化排名软件
一、基本接口或类——>DriverManager:用于管理JDBC驱动的服务类。主要功能是获取Connection对象。——>Connection:代表数据库连接对象,每个Connection代表一个物理连接会话。——>Statement:用于执行SQL语句的工具接口。…...