指針
1.&和*
在go語言中锥余,也有指針的概念宴倍,不同于java直奋。是沒有指針的概念的。但是go語言的指針也并沒有c中的復(fù)雜九杂。
首先需要了解的是指針相關(guān)的兩個符號 & 和 * 這兩種符號闽寡。
& 代表了地址操作符,表示的是取到的一個變量或者結(jié)構(gòu)的內(nèi)存地址
* 代表解地址操作尼酿,對于&類型的的變量取出它們值,如果放在類型前面則表示是指針類型植影。
func main() {
a := "a"
fmt.Println(&a)//0xc0000501f0
b := &a
fmt.Println(*b)//a
}
當(dāng)用*表示指針類型時裳擎,傳入的值需要是一個內(nèi)存地址的值,也就是傳入的值是使用&表示或者隱式指針思币。
如果改變內(nèi)存地址上的值鹿响,那么所有指向該地址的值都將被修改羡微。
func main() {
//定義一個string的指針類型,然后通過取地址b賦值給a惶我,通過解引用打印出a
var a *string
b := "a"
a = &b
fmt.Println(*a)//a
//這個時候如果修改b的值那么*a的值也將會改變妈倔,因為a存儲的是b的地址值
//所以當(dāng)b的值改變時,a通過該地址解析出的值也會是b改變了的值
b ="b"
fmt.Println(*a)//b
//當(dāng)然*a表示的也是a指向的內(nèi)存地址上的值绸贡,所以改變*a的值也相當(dāng)于改變了所有指向a地址的值盯蝴。
//因為最開始將b的地址賦值給了a,所以*a改變時听怕,b的值也發(fā)生了改變
*a = "c"
fmt.Println(b)//c
}
上面例子中a可以理解為是一個普通的變量捧挺,如果將a賦值給其它變量,那和普通變量賦值沒什么區(qū)別尿瞭,對于系統(tǒng)來說是把a地址上的值拷貝了一份給另一個變量闽烙。
2.結(jié)構(gòu)和數(shù)組的指針
對于結(jié)構(gòu)和數(shù)組,go語言提供了一些人性化的操作声搁,自動解引用黑竞。
func main() {
type skill struct{
time int
}
type people struct {
name string
age int32
skill
}
var zhangshan = &people{name: "張三"}
//兩種方式都是相同的結(jié)果,因為go語言幫我們自動解引用了疏旨,所以我們可以去除多余的括號和*
fmt.Println((*zhangshan).name)
fmt.Println(zhangshan.name)
//但是如果你要改變該地址的值很魂,相當(dāng)于在這個地址,重新new一個people充石,那么*號必不可少
*zhangshan = people{name: "李四"}
//數(shù)組也會自動解引用
num := &[8]int{0,1,2,3,4,5,6,7}
fmt.Println(num[1])
fmt.Println(num[1:4])
}
3.將指針作為參數(shù)或方法的接收者莫换、內(nèi)部指針
使用指針傳遞可以起到改變值的作用。
func main() {
one := people{
name: "張三",
age: 17,
skill:skill{
time: 10,
},
}
fmt.Println(one.age)//17
agePlusMistaken(one)//數(shù)據(jù)只是拷貝了一份過去
fmt.Println(one.age)//17
agePlus(&one)
fmt.Println(one.age)//18
//同理骤铃,使用指針類型的作為接收者時拉岁,也會起到改變值的效果
one.agePlus()
fmt.Println(one.age)//19
//對于嵌套結(jié)構(gòu)來說,go語言提供了內(nèi)部指針這種方式惰爬,為我們輕松確定結(jié)構(gòu)中指定字段的內(nèi)存地址
fmt.Println(one.time)//10
timePlus(&one.skill)
fmt.Println(one.time)//11
}
type people struct {
name string
age int32
skill
}
func (p *people) agePlus(){
p.age++
}
func agePlus(p *people){
p.age++
}
func agePlusMistaken(p people){
p.age++
}
type skill struct{
time int
}
func timePlus(s *skill){
s.time++
}
并且大家應(yīng)該注意到喊暖,上面的代碼在one并不是一個內(nèi)存地址類型。但是對于結(jié)構(gòu)體來說撕瞧,他可以自動傳入的引用或非引用參數(shù)陵叽,而不需要加&號。
但是對于內(nèi)部嵌套數(shù)據(jù)的地址丛版,則必須得通過&來進行調(diào)用
4.隱式指針
對于map來說巩掺,它本身就是一種指針,所以在傳遞的時候默認就是以指針形式傳遞页畦,對它的修改也會導(dǎo)致它原本的數(shù)據(jù)被修改胖替。(文章4說map的時候有提到過)
切片也是通過指針來設(shè)置他在一個數(shù)組里面的長度、容量和位置的。
5.指針和接口
接口也可以使用指針類型的接收者独令。
但是當(dāng)接收者為非指針時端朵,無論傳遞指針和非指針形式,對傳遞的值都不會有影響燃箭,也都可以正常調(diào)用冲呢。
但是當(dāng)接收者為指針類型時,只能傳遞指針類型的變量招狸,并且在接口內(nèi)部修改值會對外部值產(chǎn)生影響敬拓。
func main() {
p := people{
name: "張三",
age: 10,
}
hello(p)
fmt.Println(p.age)//10
hello(&p)
fmt.Println(p.age)//10
p2 := people2{
name: "張三",
age: 10,
}
//hello(p2) 當(dāng)變?yōu)橹羔槙r,只能傳遞指針類型
fmt.Println(p2.age)//10
hello(&p2)
fmt.Println(p2.age)//11
}
type speaker interface {
speak() string
}
type people struct {
name string
age int32
}
func (p people) speak() string {
p.age++
return p.name + " 說你好呀瓢颅!"
}
type people2 struct {
name string
age int32
}
func (p *people2) speak() string {
p.age++
return p.name + " 說你好呀恩尾!"
}
func hello(t speaker) {
fmt.Println(t.speak())
}