목차
Go는 일반적으로 값 전달
拷贝大的数据结构
可变性
API 一致性
表示缺失
사용해야 할까요?
백엔드 개발 Golang Go 포인터를 언제 사용해야 하는지 아시나요?

Go 포인터를 언제 사용해야 하는지 아시나요?

Sep 24, 2021 pm 03:08 PM
go 바늘

이 글은 go 언어튜토리얼 칼럼에서 제공하는 바둑 포인터(Go Pointer) 사용 상황을 소개하는 글입니다.

포인터를 사용할 때 가장 큰 오해 중 하나는 Go의 포인터가 C의 포인터와 매우 유사하다는 것입니다. 그러나 그렇지 않습니다. 포인터는 C/C++에서와 같은 방식으로 Go에서 작동하지 않습니다. Go 포인터를 언제 사용해야 하는지 아시나요?

이 글에서는 Go 포인터를 올바르게 사용하는 방법에 대해 설명합니다.

잘못된 결론: 포인터를 사용하는 것이 더 낫습니까?

일반적으로 포인터를 사용하면 값이 항상 복사되는 것을 방지하므로 애플리케이션이 더 빠르게 실행된다고 믿어집니다. Go에서도 같은 아이디어가 있다는 것은 놀라운 일이 아닙니다.

그러나 포인터 전달

Go는 일반적으로 값 전달

보다 느립니다. 이는 Go가 가비지 수집 언어이기 때문에 발생하는 결과입니다. 함수에 포인터를 전달할 때 Go는 변수가 힙에 저장되어야 하는지 아니면 스택에 저장되어야 하는지 결정하기 위해 이스케이프 분석을 수행해야 합니다. 이는 이미 약간의 추가 오버헤드를 추가하지만 그렇지 않은 경우 변수를 힙에 저장할 수 있습니다. 힙에 변수를 저장하면 GC가 실행되는 동안 시간도 손실됩니다.

Go의 편리한 기능은 go build -gcflags="-m" 명령을 실행하여 이스케이프 분석이 수행된 내용을 확인할 수 있다는 것입니다. 이렇게 하면 Go는 변수가 힙으로 이스케이프되는지 여부를 알려줍니다.

./main.go:44:20: greet ... argument does not escape
./main.go:44:21: greeting escapes to heap
./main.go:44:21: name escapes to heap
로그인 후 복사
변수가 힙으로 이스케이프되지 않으면 스택에 있습니다. 스택에는 변수를 지우는 데 가비지 수집기가 필요하지 않으며 push/pop 작업만 수행합니다. 콘텐츠가 값으로 전달되면 항상 스택에서 처리되므로 가비지 수집 오버헤드가 발생하지 않습니다. (GC는 기본적으로 실행됩니다. 힙의 콘텐츠가 적다는 것은 GC가 수행할 작업이 적다는 것을 의미합니다.)

go build -gcflags="-m" 来检查逃逸分析做了什么。如果你这样做,Go 将告诉你一个变量是否逃到堆上:

type person struct {
 name string
}func main() {
 p := person{"Richard"}
 rename(p)
 fmt.Println(p)
}func rename(p person) {
 p.name = "test"
}
로그인 후 복사

如果一个变量没有逃逸到堆中,它就在栈中。栈是不需要垃圾回收器来清除变量的,它只做 push/pop 操作。

如果任何内容都进行值传递,那么将一直在栈中做相关处理,这不会带来垃圾回收方面的开销。(GC 将按默认设置运行。堆中内容越少使得 GC 需要做的事情也越少)。

现在你知道了吧,使用指针反而会降低性能,那么什么时候需要使用指针呢?

拷贝大的数据结构

指针是否一直表现的比值传递差呢?显然不是这样的。对大的数据结构进行处理时,指针将发挥作用。这样可能会使得垃圾回收的开销被拷贝大量数据的开销抵消掉。

当我提到这点时,总是被问到‘那个大数据应该多大’?

我觉得这里没有一个固定的数值,凡是与性能相关的,都应该对其进行基准测试。 Go 有内置的强大的基准测试工具,完全可以利用起来  

可变性

唯一能修改函数参数的方式是传指针。默认对值的修改都是在副本上进行的。因此这些修改不能在调用它的函数中体现。

看下面的代码:

func main() {
 p := person{"Richard"}
 rename(&p)
 fmt.Println(p)
}func rename(p *person) {
 p.name = "test"
}
로그인 후 복사

输出是 Richard ,因为对 person 的修改是在它的副本上进行的。如果要改变底层 person 对象的值,需要使用指针。

func (p *person) rename(s string) {
   p.name = s 
}func (p *person) printName() {
  fmt.Println(p.name)
}
로그인 후 복사

如上,输出 test 。可变性是指针在 Go 中使用的一种情景。这是否是好事,还需要讨论。

API 一致性

使用指针可以维持最新值。这可以保持 API 一致性,即使不是所有的方法都改变它的值。

因此,这个:

func (p *person) rename(s string) {
   p.name = s 
}func (p person) printName() {
  fmt.Println(p.name)
}
로그인 후 복사

优于

type exam struct {
    score   int
    present bool
}
로그인 후 복사

虽然为了一致性并不需要在 printName 中使用指针。但是这将使得 API 更简单,避免去记到底哪里需要引用。

表示缺失

一般值在使用时,具有默认零值。但有些情景需要知道某个事物是缺少或未填充值。例如一个结构体包含学生的考试分数,如果结构体是空且有分数 0 ,这表示这个学生考的不好,还是压根没有参加考试呢?

指针的默认零值是 nil 指针,表示没有设置值。也可以像下面这样实现这种要求:

func main() {
 p := person{"richard"}
 p = rename(p)
 fmt.Println(p)
}func rename(p person) person {
 p.name = "test"
 return p
}
로그인 후 복사
로그인 후 복사

使用单独的 present이제 아시다시피 포인터를 사용하면 성능이 저하되므로 언제 포인터를

필요

사용해야 할까요?

대규모 데이터 구조 복사포인터는 항상 값 전송보다 성능이 떨어지나요? 분명히 그렇지 않습니다. 대규모 데이터 구조로 작업할 때 포인터가 작동합니다. 이로 인해 가비지 수집 비용이 대량의 데이터 복사 비용으로 상쇄될 수 있습니다.

이 얘기를 하면 항상 '그 빅데이터는 얼마나 커야 하는가?'라는 질문을 받습니다.

여기에는 고정된 값은 없고 성능과 관련된 모든 것을 벤치마킹해야 한다고 생각합니다. Go에는 완전히 활용할 수 있는 강력한 벤치마킹 도구가 내장되어 있습니다.

🎜Variability🎜🎜함수 매개변수를 수정하는 유일한 방법은 포인터를 전달하는 것입니다. 기본적으로 값 수정은 복사본에서 수행됩니다. 따라서 이러한 수정 사항은 이를 호출하는 함수에 반영될 수 없습니다. 🎜🎜아래 코드를 보세요: 🎜
x := []int{1,2}
x = append(x, 3)
x = append(x, 4)
로그인 후 복사
로그인 후 복사
🎜person에 대한 수정 사항이 복사본에 적용되었기 때문에 출력은 Richard입니다. 기본 사람 개체의 값을 변경하려면 포인터를 사용해야 합니다. 🎜rrreee🎜 위와 같이 test를 출력합니다. 가변성은 Go에서 포인터가 사용되는 상황입니다. 이것이 좋은 것인지 여부는 논쟁의 여지가 있습니다. 🎜🎜API 일관성🎜🎜포인터를 사용하여 최신 값을 유지하세요. 이는 모든 메소드가 값을 변경하지 않더라도 API의 일관성을 유지합니다. 🎜🎜따라서 🎜rrreee🎜는 🎜rrreee🎜보다 낫습니다. 하지만 일관성을 위해 printName에서 포인터를 사용할 필요는 없습니다. 그러나 이렇게 하면 API가 더 단순해지고 참조가 필요한 정확한 위치를 기억할 필요가 없습니다. 🎜🎜은 누락을 나타냅니다. 🎜🎜일반 값은 사용 시 기본값이 0입니다. 그러나 무언가가 누락되었거나 채워지지 않은 값이 있음을 알아야 하는 시나리오가 있습니다. 예를 들어, 구조에 학생의 시험 점수가 포함되어 있고 구조가 비어 있고 점수가 0이면 학생이 시험을 잘 치르지 않았거나 시험을 전혀 치지 않았다는 의미입니까? 🎜🎜포인터의 기본 0 값은 nil 포인터입니다. 이는 값이 설정되지 않았음을 의미합니다. 이 요구 사항은 다음과 같이 구현할 수도 있습니다. 🎜rrreee🎜별도의 present 필드를 사용하여 학생이 시험에 응시하지 않았음을 나타냅니다. 🎜🎜나는 왜 가치를 선택했나요? 🎜🎜🎜다소 주관적입니다. 사람들마다 프로그래밍에 대한 이해가 다르기 때문에 모든 사람이 동일한 개념을 가질 필요는 없습니다. Go 값에는 가능한 한 기본값을 갖는 것이 합리적이라고 생각합니다. 모든 경우에 적용되는 것은 아니지만 제 경우에는 큰 사고를 예방할 수 있었습니다. 포인터 대신 값을 사용해도 널 포인터로 인한 Tony Hoare의 "백만 달러 실수"가 발생하지 않습니다. 🎜🎜기본값 0은 많은 선언을 피하는 데 유용합니다. 🎜

另一个好处是易变性造成的问题比它解决的问题多的得多。易变性给函数带来的副作用同时使得调试变得更加困难。 通过让函数返回修改之后的结构体,可以避免这种突变。

重写之前的例子

func main() {
 p := person{"richard"}
 p = rename(p)
 fmt.Println(p)
}func rename(p person) person {
 p.name = "test"
 return p
}
로그인 후 복사
로그인 후 복사

这也是 append 如何工作的,所以并不陌生。

x := []int{1,2}
x = append(x, 3)
x = append(x, 4)
로그인 후 복사
로그인 후 복사

鉴于指针的安全性,和值处理比指针处理更快,使用指针需要反复斟酌。

原文地址:https://medium.com/@meeusdylan/when-to-use-pointers-in-go-44c15fe04eac

译文地址:https://learnku.com/go/t/60923

위 내용은 Go 포인터를 언제 사용해야 하는지 아시나요?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

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

뜨거운 기사 태그

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

Go WebSocket 메시지를 보내는 방법은 무엇입니까? Go WebSocket 메시지를 보내는 방법은 무엇입니까? Jun 03, 2024 pm 04:53 PM

Go WebSocket 메시지를 보내는 방법은 무엇입니까?

Golang 기술 성능 최적화에서 메모리 누수를 방지하는 방법은 무엇입니까? Golang 기술 성능 최적화에서 메모리 누수를 방지하는 방법은 무엇입니까? Jun 04, 2024 pm 12:27 PM

Golang 기술 성능 최적화에서 메모리 누수를 방지하는 방법은 무엇입니까?

Go에서 정규식을 사용하여 타임스탬프를 일치시키는 방법은 무엇입니까? Go에서 정규식을 사용하여 타임스탬프를 일치시키는 방법은 무엇입니까? Jun 02, 2024 am 09:00 AM

Go에서 정규식을 사용하여 타임스탬프를 일치시키는 방법은 무엇입니까?

Golang과 Go 언어의 차이점 Golang과 Go 언어의 차이점 May 31, 2024 pm 08:10 PM

Golang과 Go 언어의 차이점

C++의 포인터 및 참조에 대한 심층 분석을 통해 메모리 사용 최적화 C++의 포인터 및 참조에 대한 심층 분석을 통해 메모리 사용 최적화 Jun 02, 2024 pm 07:50 PM

C++의 포인터 및 참조에 대한 심층 분석을 통해 메모리 사용 최적화

Golang 프레임워크 문서화 모범 사례 Golang 프레임워크 문서화 모범 사례 Jun 04, 2024 pm 05:00 PM

Golang 프레임워크 문서화 모범 사례

단위 테스트 Go 동시 기능 가이드 단위 테스트 Go 동시 기능 가이드 May 03, 2024 am 10:54 AM

단위 테스트 Go 동시 기능 가이드

Go에서 우선순위가 높은 고루틴을 만드는 방법은 무엇입니까? Go에서 우선순위가 높은 고루틴을 만드는 방법은 무엇입니까? Jun 04, 2024 pm 12:41 PM

Go에서 우선순위가 높은 고루틴을 만드는 방법은 무엇입니까?

See all articles