T
做爲參數類型,何時選擇 *T
做爲參數類型?[ ] T
是傳遞的指針仍是值?選擇 [ ] T
仍是 [ ] *T
?先看傳 T 的狀況:數組
type user struct { id int name string } func passByValue(_u user){ _u.id++ _u.name="jack" // when printing structs, the plus flag (%+v) adds field names fmt.Printf("_u 值:%+v;地址:%p; \n",_u,&_u) } func exp2(){ u:=user{1,"peter"} fmt.Printf("原始 u 值:%+v; 地址: %p;\n",u,&u) passByValue(u) fmt.Printf("執行完函數後 u 值:%+v; 地址: %p;\n",u,&u) }
執行 exp2 方法,輸出結果爲:
結果說明:函數
再看傳 *T 的狀況:性能
type user struct { id int name string } func passByPointer(_u *user){ _u.id++ _u.name="jack" fmt.Printf("_u 值:%+v ;u指向的地址:%p; u自己存放地址:%p; \n",*_u,_u,&_u) } func exp3(){ u:=&user{1,"peter"} fmt.Printf("原始u 值:%+v; 指向的地址: %p;u自己存放地址: %p; \n",*u,u,&u) passByPointer(u) fmt.Printf("原始u 值:%+v; 指向的地址: %p;u自己存放地址: %p; \n",*u,u,&u) }
執行 exp3 方法的輸出結果爲:spa
注意到,雖然參數 _u 仍然是 u 的一份拷貝對象,可是原始對象的值仍是改變了。能夠這麼理解,由於 u 指針和 _u 指針都指向同一個對象,即 0xc0000484a0 地址上存放的對象,_u.name="jack"
能夠看作*(_u).name="jack
,即取值後再改變值。指針
type user struct { id int name string } func changeAddress(_u *user){ _u=&user{2,"jack"} fmt.Printf("參數_u 值:%+v ;u指向的地址:%p; u自己存放地址:%p; \n",*_u,_u,&_u) return } func exp4(){ u:=&user{1,"peter"} fmt.Printf("原始u 值:%+v; 指向的地址: %p;u自己存放地址: %p; \n",*u,u,&u) changeAddress(u) fmt.Printf("執行函數後 u 值:%+v; 指向的地址: %p;u自己存放地址: %p; \n",*u,u,&u) }
輸出結果以下:code
注意,執行函數後 u 值沒有改變!改變了參數指向的地址,原來的對象確定就不受影響了。對象
func passSlice(_s []int){ _s[0]=99 fmt.Printf("_s 值:%v,地址:%p\n",_s,&_s) } func exp6(){ s:=[]int{11,22,33,44} fmt.Printf("s 值:%v,地址:%p\n",s,&s) passSlice(s) fmt.Printf("執行函數後 s 值:%v,地址:%p\n",s,&s) }
對切片參數的修改會影響原來的切片。blog
再看傳遞數組索引
func passArray(_a [3]int){ _a[0]=99 fmt.Printf("_a 值:%v,地址:%p\n",_a,&_a) } func exp7(){ a:=[3]int{22,33,44} fmt.Printf("a 值:%v,地址:%p\n",a,&a) passArray(a) fmt.Printf("執行函數後 a 值:%v,地址:%p\n",a,&a) }
對數組參數的修改並不會影響原來的切片。ip
u1:=u2
。包括 slice,map,array 在初始化和按索引設置的時候都會建立副本對函數的參數或者返回值定義成 T 仍是 *T 要考慮如下幾點:
參考文章出處:
https://colobu.com/2017/01/05...