Go 언어의 defer는 적어도 오늘날의 주류 프로그래밍 언어와 비교하면 언어의 새로운 기능으로 간주됩니다.
defer 문은 함수를 호출합니다. 주변 함수가 반환되거나 주변 함수가 끝까지 실행되거나 해당 goroutine 패닉이 발생할 때까지 이 함수의 실행이 연기됩니다.
defer가 실행될 때마다 그 뒤에 있는 함수 값( in go 함수는 참조 유형, 일급 시민이며 변수에 할당될 수 있음) 및 함수 매개 변수가 평가되지만 위의 세 가지 상황이 (↑) 발생할 때까지 함수가 즉시 호출되지 않습니다. 이것이 defer의 전체 내용입니다. 더 이상은 필요하지 않습니다. 나머지는 defer의 모범 사례입니다.
함수는 즉시 호출되지 않습니다
가장 간단한 것부터 시작해 보겠습니다.
func readFile(fileName string){ f,err := os.Open(fileName) if err!=nil { return } defer f.Close() var content [1024]byte f.Read(content[:]) fmt.Printf("%s",content) } func main() { readFile("test.data") }
프로그램은 defer의 처음 1024바이트를 출력합니다. 테스트.데이터. 이와 같은 열기/닫기 페어링 작업이 defer의 관용적 사용법이라는 점은 언급할 가치가 있습니다. 이 예는 위 문장의 후반부를 설명합니다
"그러나 함수는 호출되지 않습니다"
defer 후 f.Close()가 지연되지 않으면 파일 설명자가 닫히고 읽을 수 없기 때문입니다. 어떤 콘텐츠.
함수 값과 함수 매개변수가 평가되지만 함수가 즉시 호출되지 않습니다.
다음 예는 전반부를 설명합니다. <>에서 비롯되며 약간 수정되었습니다.
func trace(funcName string) func(){ start := time.Now() fmt.Printf("function %s enter\n",funcName) return func(){ log.Printf("function %s exit (elapsed %s)",funcName,time.Since(start)) } } func foo(){ defer trace("foo()")() time.Sleep(5*time.Second) } func main(){ foo() foo() } /* OUTPUT: function foo() enter function foo() exit (elapsed 5.0095471s) function foo() enter function foo() exit (elapsed 5.0005382s) */
foo 출력이 입력되는 이유 그런 다음 5초 정도 기다렸다가 종료를 출력하시겠습니까? 앞서 말했듯이
defer 이후의 함수 값과 매개변수는 평가되지만 실제 함수 호출은 끝까지 기다려야 합니다.
여기서 함수 값은 반환된 익명 함수입니다. Trace()에 의해 함수 매개변수는 물론 문자열 리터럴 "foo()"입니다. Trace("foo()")를 평가하면 foo() Enter 함수가 출력됩니다. ")()는 foo 함수를 출력합니다. () exit(elapsed x.x)는 반환 실행까지 연기됩니다(return이 반환 값 변수를 업데이트하는 경우 defer 함수는 업데이트 후에만 실행됩니다).
기타
조금 더, defer 문이 여러 개 있는 경우 마지막 defer 함수의 실행 순서는 defer가 나타나는 순서와 반대입니다. 예:
func main() { func1 := func(){ fmt.Println("func1() execution deferred") } func2 := func(){ fmt.Println("func2() execution deferred") } defer func1() defer func2() fmt.Println("strat\nworking...") } /* OUTPUT: strat working... func2() execution deferred func1() execution deferred */
위 내용은 go defer 소개(go Delay 기능)의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!