Ответ 1
Аргументы функции Go передаются по значению.
Во-первых, отбросьте нерелевантные части вашего примера, чтобы мы могли легко видеть, что вы просто передаете аргумент по значению. Например,
package main
import "fmt"
func byval(q *int) {
fmt.Printf("3. byval -- q %T: &q=%p q=&i=%p *q=i=%v\n", q, &q, q, *q)
*q = 4143
fmt.Printf("4. byval -- q %T: &q=%p q=&i=%p *q=i=%v\n", q, &q, q, *q)
q = nil
}
func main() {
i := int(42)
fmt.Printf("1. main -- i %T: &i=%p i=%v\n", i, &i, i)
p := &i
fmt.Printf("2. main -- p %T: &p=%p p=&i=%p *p=i=%v\n", p, &p, p, *p)
byval(p)
fmt.Printf("5. main -- p %T: &p=%p p=&i=%p *p=i=%v\n", p, &p, p, *p)
fmt.Printf("6. main -- i %T: &i=%p i=%v\n", i, &i, i)
}
Вывод:
1. main -- i int: &i=0xf840000040 i=42
2. main -- p *int: &p=0xf8400000f0 p=&i=0xf840000040 *p=i=42
3. byval -- q *int: &q=0xf8400000d8 q=&i=0xf840000040 *q=i=42
4. byval -- q *int: &q=0xf8400000d8 q=&i=0xf840000040 *q=i=4143
5. main -- p *int: &p=0xf8400000f0 p=&i=0xf840000040 *p=i=4143
6. main -- i int: &i=0xf840000040 i=4143
В функции main
, i
является переменной int
в ячейке памяти (&i
) 0xf800000040
с начальным значением (i
) 42
.
В функции main
, p
является указателем на переменную int
в ячейке памяти (&p
) 0xf8000000f0
со значением (p
= &i
) 0xf800000040
, которое указывает на значение int
(*p
= i
) 42
.
В функции main
, byval(p)
- вызов функции, который присваивает значение (p
= &i
) 0xf800000040
аргумента в ячейке памяти (&p
) 0xf8000000f0
функции byval
параметр q
в ячейке памяти (&q
) 0xf8000000d8
. Другими словами, память выделяется для параметра byval
q
, и ему присваивается значение аргумента main
byval
p
; значения p
и q
вначале совпадают, но переменные p
и q
различны.
В функции byval
, используя указатель q
(*int
), который является копией указателя p
(*int
), integer *q
(i
) устанавливается в новый int значение 4143
. В конце перед возвращением. указатель q
установлен на nil
(нулевое значение), который не влияет на p
, так как q
является копией.
В функции main
, p
является указателем на переменную int
в ячейке памяти (&p
) 0xf8000000f0
со значением (p
= &i
) 0xf800000040
, которое указывает на новое значение int
(*p
= i
) 4143
.
В функции main
, i
является переменной int
в ячейке памяти (&i
) 0xf800000040
с конечным значением (i
) 4143
.
В вашем примере функция main
variable s
, используемая в качестве аргумента для вызова функции gotest
, не совпадает с параметром gotest
s
. Они имеют одно и то же имя, но представляют собой разные переменные с различными областями и ячейками памяти. Параметр функции s
скрывает аргумент вызова функции s
. Вот почему в моем примере я назвал аргументы и переменные параметра p
и q
соответственно, чтобы подчеркнуть разницу.
В вашем примере (&s
) 0x4930d4
- адрес ячейки памяти для переменной s
в функции main
, которая используется в качестве аргумента для вызова функции gotest(s, done)
и 0x4974d8
- адрес ячейки памяти для параметра gotest
функции s
. Если вы установите параметр s = nil
в конце функции gotest
, он не влияет на переменную s
в main
; s
в main
и s
в gotest
- различные ячейки памяти. В терминах типов &s
- **Something
, s
- *Something
, а *s
- Something
. &s
- это указатель на адрес (адрес ячейки памяти) s
, который является указателем на (адрес ячейки памяти) анонимной переменной типа Something
. В терминах значений main.&s != gotest.&s
, main.s == gotest.s
, main.*s == gotest.*s
и main.s.number == gotest.s.number
.
Вы должны принять совет mkb sage и прекратить использование println(&s)
. Используйте пакет fmt
, например,
fmt.Printf("%v %p %v\n", &s, s, *s)
Указатели имеют одинаковое значение, когда указывают на одну и ту же ячейку памяти; указатели имеют разные значения, когда указывают на разные ячейки памяти.