> 백엔드 개발 > Golang > Go 언어에서 지연과 패닉의 관계는 무엇입니까?

Go 언어에서 지연과 패닉의 관계는 무엇입니까?

王林
풀어 주다: 2023-06-11 09:00:08
원래의
950명이 탐색했습니다.

Go 언어는 몇 가지 독특한 구문과 기능을 추가한 새로운 프로그래밍 언어입니다. 그 중 지연과 패닉은 매우 중요한 두 가지 기능입니다. 이번 글에서는 Go 언어에서 defer와 Panic의 관계와 사용법, 특징을 소개하겠습니다.

defer 사용법

Go 언어의 defer 문은 함수의 실행이 끝나거나 현재 범위가 끝나면 등록된 함수가 자동으로 실행됩니다. defer는 리소스 릴리스, 잠금 잠금 해제 및 오류 처리와 같은 여러 시나리오에서 사용될 수 있습니다.

다음은 리소스 지연 해제 예시입니다.

func main() {
    file, err := os.Open("myfile.txt")
  // 在函数结束时,会自动关闭文件
    defer file.Close()
  
    if err != nil {
        fmt.Println("Failed to open file.")
        return
    }
    // ...
}
로그인 후 복사

는 defer 파일의 Close() 함수를 통해 등록되며, 함수 실행이 끝나면 파일이 자동으로 닫힙니다.

다음은 연기 잠금 해제의 예입니다.

func main() {
    var lock sync.Mutex
    lock.Lock()
  // 当函数执行结束时,会自动解锁
    defer lock.Unlock()
  
    // ...
}
로그인 후 복사

함수 실행이 끝나면 Unlock() 함수가 자동으로 호출되어 잠금을 해제합니다.

defer의 실행 순서는 뒤에서 앞으로 진행되는데, 이는 defer 문이 여러 개 등록되면 역순으로 실행된다는 의미입니다. 다음 예에서는 Defer 2를 출력한 다음 Defer 1을 출력합니다. Defer 2,然后输出Defer 1

func main() {
    defer fmt.Println("Defer 1")
    defer fmt.Println("Defer 2")
    fmt.Println("Hello")
}
로그인 후 복사

panic 的用法

Go 语言中的 panic 关键字用于抛出一个异常,并终止当前函数或者程序的执行。 panic 会沿着函数调用堆栈向上传递,直到被 recover() 函数捕捉到为止。如果没有被捕捉到,整个程序会被退出,并输出一个调用堆栈。

下面示例代码中,当输入的字符串长度小于 5 时,会触发 panic,终止程序的执行。

func hello(name string) {
    if len(name) < 5 {
        panic("Name is too short.")
    }
    fmt.Println("Hello", name)
}

func main() {
    hello("Tom")
    hello("Bob")
    hello("me")
}
로그인 후 복사

输出结果如下:

Hello Tom
Hello Bob
panic: Name is too short.

goroutine 1 [running]:
main.hello(...)
    /Users/user/goland/src/main.go:4
main.main()
    /Users/user/goland/src/main.go:10 +0x81
exit status 2
로그인 후 복사

这里我们可以看到当输入的 name 为me

func main() {
    defer fmt.Println("Defer 1")
    defer fmt.Println("Defer 2")
    
    panic("Oops! Something went wrong.")
}
로그인 후 복사

panic Usage

Go 언어의 패닉 키워드는 예외를 발생시키고 현재 함수나 프로그램의 실행을 종료하는 데 사용됩니다. 패닉은 Recover() 함수에 의해 포착될 때까지 함수 호출 스택 위로 전파됩니다. 잡히지 않으면 전체 프로그램이 종료되고 호출 스택이 인쇄됩니다.

아래 예제 코드에서는 입력 문자열의 길이가 5보다 작을 경우 패닉이 발생하고 프로그램 실행이 종료됩니다.

Defer 2
Defer 1
panic: Oops! Something went wrong.

goroutine 1 [running]:
main.main()
    /Users/user/goland/src/main.go:7 +0x81
exit status 2
로그인 후 복사

출력 결과는 다음과 같습니다.

func init() {
    fmt.Println("Init 1")
}

func init() {
    fmt.Println("Init 2")
}

func main() {
    defer fmt.Println("Defer 1")
    
    defer func() {
        if err := recover(); err != nil {
            fmt.Println("Recovered:", err)
        }
    }()
    
    defer fmt.Println("Defer 2")
    
    panic("Oops! Something went wrong.")
}
로그인 후 복사

여기서 입력 이름이 me이면 패닉이 발생하고 프로그램 실행이 종료되는 것을 볼 수 있습니다.

defer와 패닉의 관계

panic은 프로그램 실행을 즉시 종료하는 역할을 하며, 이는 함수 실행이 끝나기 전을 포함하여 언제든지 트리거될 수 있음을 의미합니다. 프로그램이 제때에 리소스를 해제하고 필요한 일부 정리 작업을 수행할 수 있도록 하기 위해 Go 언어는 함수가 종료되기 전에 일부 정리 작업을 수행할 수 있도록 하는 연기 메커니즘을 도입합니다.

함수에서 패닉이 발생하면 즉시 종료되고 현재 함수 이전에 등록된 모든 연기 함수가 실행됩니다. 다음 예제 코드는 패닉을 수동으로 트리거하고 종료하기 전에 defer 함수를 두 번 실행합니다.

Init 1
Init 2
Defer 2
Recovered: Oops! Something went wrong.
Defer 1
로그인 후 복사
출력 결과는 다음과 같습니다.

rrreee

패닉이 발생한 후 두 개의 defer 함수가 역순으로 실행되는 것을 볼 수 있습니다.

함수 끝에 연기 함수를 등록하는 것 외에도 Go 언어에서는 여러 연기 함수를 함수에 등록할 수 있습니다. 이는 함수에 여러 개의 연기 기능이 있는 경우 연기 중 하나가 패닉을 유발하더라도 다른 연기 기능은 계속 실행될 수 있음을 의미합니다. 🎜🎜다음 예제 코드는 함수가 여러 defer 문을 등록할 때 defer 함수 중 하나가 패닉을 유발하지만 다른 defer 함수는 계속 실행된다는 것을 보여줍니다. 🎜rrreee🎜출력 결과는 다음과 같습니다. 🎜rrreee🎜함수는 먼저 두 개의 init 함수를 실행한 다음 세 개의 defer 함수를 순차적으로 실행하는 것을 볼 수 있습니다. 지연 중 하나가 패닉을 유발하지만 다른 지연에 의해 포착되어 프로그램이 재개되면 결국 두 지연 기능이 모두 정상적으로 실행됩니다. 🎜🎜실제 개발에서는 defer와 패닉이 쌍으로 사용되는 경우가 많습니다. defer는 리소스를 해제하고 정리 작업을 수행하는 데 사용되고, 패닉은 예외를 처리하는 데 사용됩니다. 함수가 종료되기 전에 여러 정리 작업을 수행해야 하는 경우 함수 시작 부분에 defer 래퍼 함수를 ​​사용하고 복구() 함수를 사용하여 함수가 조기에 종료되는 것을 방지할 수 있습니다. 이러한 작성 방식은 매우 일반적이며 강력한 프로그램을 작성하는 데 강력한 보장을 제공합니다. 🎜

위 내용은 Go 언어에서 지연과 패닉의 관계는 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

관련 라벨:
원천:php.cn
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
최신 이슈
인기 튜토리얼
더>
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿