>  기사  >  백엔드 개발  >  Go 언어의 포인터에 대한 연산은 무엇입니까?

Go 언어의 포인터에 대한 연산은 무엇입니까?

青灯夜游
青灯夜游원래의
2023-01-04 10:30:293545검색

Go 언어에는 포인터 연산이 없습니다. Go 언어의 구문은 포인터 연산을 지원하지 않으며 모든 포인터는 제어 가능한 범위 내에서 사용되지만 실제로 Go 언어는 unsafe 패키지의 Pointer() 메서드를 통해 간접적으로 포인터를 uintptr 유형 숫자로 변환할 수 있습니다. 포인터 연산을 구현합니다.

Go 언어의 포인터에 대한 연산은 무엇입니까?

이 튜토리얼의 운영 환경: Windows 7 시스템, GO 버전 1.18, Dell G3 컴퓨터.

포인터란 정확히 무엇인가요?

메모리는 일련번호가 있는 일련의 저장 단위입니다. 변수는 컴파일러가 메모리 주소에 할당한 별명입니다.

포인터는 다른 메모리 주소 변수를 가리키는 값입니다

포인터는 변수의 메모리 주소를 가리키며, 포인터는 변수 값의 메모리 주소와 같습니다

코드를 살펴보겠습니다 snippet

func main() {
    a := 200
    b := &a
    *b++
    fmt.Println(a)
}

주 함수 첫 번째 줄에서는 새 변수 a를 정의하고 값 200을 할당합니다. 다음으로 변수 b를 정의하고 변수 a의 주소를 b에 할당합니다. 우리는 a의 정확한 저장 주소를 모르지만 변수 b에 a의 주소를 저장할 수 있습니다.

Go 언어의 포인터에 대한 연산은 무엇입니까?

Go 언어의 포인터에 대한 연산은 무엇입니까?

Go의 강력한 형식 특성으로 인해 코드의 세 번째 줄은 아마도 가장 방해가 될 것입니다. b에는 a 변수의 주소가 포함되어 있지만 우리는 a 변수에 저장된 값을 증가시키려고 합니다. .

이 방법으로 b를 역참조하고 대신 포인터 다음에 b를 사용하여 a를 참조해야 합니다.
그런 다음 값에 1을 더하고 b에 저장된 메모리 주소에 다시 저장합니다.

마지막 줄은 a의 값이 201

Go 언어의 포인터에 대한 연산은 무엇입니까?

Go 언어의 포인터에 대한 연산은 무엇입니까?

우리가 원할 때의 모든 값 복사본입니다. 수정하려면 특정 변수를 사용하면 해당 변수의 주소를 가리키는 포인터 변수를 생성할 수 있습니다.

C/C++의 포인터와 달리

Go 언어의 포인터는 오프셋 및 연산이 불가능하며 안전한 포인터입니다.

Go 언어의 포인터를 이해하려면 먼저 3가지 개념을 알아야 합니다:

포인터 주소, 포인터 유형 및 포인터 값.

포인터 주소 및 포인터 유형

Go 언어의 포인터 연산은 매우 간단합니다: &(주소 가져오기) 및 *(주소 기반 값 가져오기) 두 가지 기호만 기억하면 됩니다.

각 변수에는 런타임 시 주소가 있으며, 이 주소는 메모리에서 변수의 위치를 ​​나타냅니다. Go 언어에서는 변수 앞에 & 문자를 사용하여 변수의 "주소를 가져옵니다".

변수 포인터를 가져오는 구문은 다음과 같습니다.

ptr := &v    // v的类型为T

그 중

  • v: 가져오는 주소를 나타내는 변수, 유형은 T

  • ptr: 가져오는 데 사용되는 변수 주소에서 ptr의 유형은 * T입니다. 이를 T의 포인터 유형이라고 합니다. *는 포인터를 의미합니다.

Go 언어의 포인터에 대한 연산은 무엇입니까?

func main() {
    a := 10
    b := &a
    fmt.Printf("a:%d ptr:%p\n", a, &a) // a:10 ptr:0xc00001a078
    fmt.Printf("b:%p type:%T\n", b, b) // b:0xc00001a078 type:*int
    fmt.Println(&b)                    // 0xc00000e018
}

포인터 연산자

1. 포인터 연산자가 lvalue인 경우 대상 개체의 상태를 업데이트할 수 있습니다. 이는 대상 개체의 상태를 가져오는 것입니다. .

func main() {
    x := 10
    var p *int = &x  //获取地址,保存到指针变量
    *p += 20        //用指针间接引用,并更新对象
    println(p, *p)  //输出指针所存储的地址,以及目标对象
}
출력:

0xc000040780 30

2. 포인터 유형은 동등 연산자를 지원하지만 더하기, 빼기 및 유형 변환을 수행할 수 없습니다. 두 포인터가 동일한 주소를 가리키거나 둘 다 nil이면 동일합니다.

func main() {
    x := 10
    p := &x

    p++   //编译报错 invalid operation: p++ (non-numeric type *int)
    var p2 *int = p+1  //invalid operation: p + 1 (mismatched types *int and int)
    p2 = &x
    println(p == p2)   //指向同一地址
}
unsafe.Pointer를 사용하여 포인터를 uintptr로 변환한 후 덧셈, 뺄셈 연산을 수행할 수 있지만 불법 접근이 발생할 수 있습니다.

포인터 연산

많은 golang 프로그램에서는 포인터를 사용하더라도 포인터를 더하거나 빼지 않는다는 점에서 C 프로그램과 많이 다릅니다. Golang의 공식 입문 학습 도구(

go Tour)에는 Go가 포인터 연산을 지원하지 않는다고까지 나와 있습니다. 실제로는 그렇지는 않지만 일반 Go 프로그램에서 포인터 연산을 본 적이 없는 것 같습니다(글쎄, 특이한 프로그램을 작성하고 싶어한다는 것을 알고 있습니다).

  • 但实际上,go 可以通过 unsafe.Pointer 来把指针转换为 uintptr 类型的数字,来间接实现指针运算。
  • 这里请注意,uintptr 是一种整数类型,而不是指针类型。

比如:

uintptr(unsafe.Pointer(&p)) + 1

就得到了 &p 的下一个字节的位置。然而,根据 《Go Programming Language》 的提示,我们最好直接把这个计算得到的内存地址转换为指针类型:

unsafe.Pointer(uintptr(unsafe.Pointer(&p) + 1))

因为 go 中是有垃圾回收机制的,如果某种 GC 挪动了目标值的内存地址,以整型来存储的指针数值,就成了无效的值。

同时也要注意,go 中对指针的 + 1,真的就只是指向了下一个字节,而 C 中 + 1 或者 ++ 考虑了数据类型的长度,会自动指向当前值结尾后的下一个字节(或者说,有可能就是下一个值的开始)。如果 go 中要想实现同样的效果,可以使用 unsafe.Sizeof 方法:

unsafe.Pointer(uintptr(unsafe.Pointer(&p) + unsafe.Sizeof(p)))

最后,另外一种常用的指针操作是转换指针类型。这也可以利用 unsafe 包来实现:

var a int64 = 1
(*int8)(unsafe.Pointer(&a))

如果你没有遇到过需要转换指针类型的需求,可以看看这个项目(端口扫描工具),其中构建 IP 协议首部的代码,就用到了指针类型转换。

【相关推荐:Go视频教程编程教学

위 내용은 Go 언어의 포인터에 대한 연산은 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.