> 백엔드 개발 > Golang > Go에서 리소스를 적절하게 해제하려면 루프에서 `defer` 문을 어디에 배치해야 합니까?

Go에서 리소스를 적절하게 해제하려면 루프에서 `defer` 문을 어디에 배치해야 합니까?

Susan Sarandon
풀어 주다: 2025-01-02 17:21:40
원래의
605명이 탐색했습니다.

Where Should I Place `defer` Statements in a Loop to Properly Release Resources in Go?

루프에서 연기를 사용하여 리소스를 해제하는 방법: 최선의 접근 방식

루프 내 데이터베이스 쿼리와 관련된 상황에서 다음과 같은 질문이 발생합니다. 적절한 리소스 릴리스를 보장하기 위해 지연 문을 최적으로 배치합니다. 다음 루프를 고려하십시오.

for rows.Next() {

   fields, err := db.Query(.....)
   if err != nil {
      // ...
   }
   defer fields.Close()

   // do something with `fields`

}
로그인 후 복사

배치 연기를 위한 두 가지 옵션이 나타납니다.

  1. 루프 본문 내에서 연기:

    for rows.Next() {
    
       fields, err := db.Query(.....)
       if err != nil {
          // ...
       }
    
       // do something with `fields`
    }
    
    defer fields.Close()
    로그인 후 복사
  2. 루프 이후 연기 body:

    for rows.Next() {
    
       fields, err := db.Query(.....)
       if err != nil {
          // ...
       }
       
       // do something with `fields`
       
       defer fields.Close()
    }
    로그인 후 복사

지연 실행 이해

최상의 접근 방식을 선택하려면 먼저 지연 동작을 이해해야 합니다. 지연된 함수는 주변 함수가 반환될 때까지 지연될 뿐만 아니라, 예외(예: 패닉)로 인해 함수가 갑자기 종료되는 경우에도 실행됩니다. 이는 예외적인 상황에서도 리소스 해제를 보장하는 필수 메커니즘 역할을 합니다.

배치 연기의 잠재적인 문제

루프 내부에 연기를 배치하면 루프 종료 시 리소스 해제가 방해될 수 있습니다. . 루프 내에서 처리된 오류로 인해 루프가 일찍 종료되는 경우 deferred fields.Close() 호출은 실행되지 않습니다.

반대로 루프 본문 뒤에 defer를 배치하면 리소스가 어떻게 실행되는지에 관계없이 해제됩니다. 루프가 종료됩니다. 그러나 이 접근 방식은 전체 루프가 완료될 때까지 리소스 정리를 지연시키며 이는 모든 시나리오에서 바람직하지 않을 수 있습니다.

최적의 솔루션: 익명 또는 명명된 함수 래퍼

두 가지 문제를 모두 해결하려면 익명 또는 명명된 함수 내에서 리소스 할당 및 해제를 캡슐화하는 것이 좋습니다. 이렇게 하면 함수 내에서 defer를 사용하여 함수 반환 시 리소스가 해제되도록 할 수 있습니다.

예:

// Anonymous function wrapper
for rows.Next() {
    func() {
        fields, err := db.Query(...)
        if err != nil {
            // Handle error and return
            return
        }
        defer fields.Close()

        // do something with `fields`
    }()
}

// Named function wrapper
func foo(rs *db.Rows) {
    fields, err := db.Query(...)
    if err != nil {
        // Handle error and return
        return
    }
    defer fields.Close()

    // do something with `fields`
}

for rows.Next() {
    foo(rs)
}
로그인 후 복사

이 접근 방식을 사용하면 리소스가 더 이상 존재하지 않는 즉시 리소스를 해제할 수 있습니다. 예외적인 경우에도 필요합니다. 또한 목표가 첫 번째 오류에서 루프를 종료하는 것이라면 래퍼 함수에서 오류를 반환하고 그에 따라 처리할 수 있습니다.

func foo(rs *db.Rows) error {
    fields, err := db.Query(...)
    if err != nil {
        return fmt.Errorf("db.Query error: %w", err)
    }
    defer fields.Close()

    // do something with `fields`
    return nil
}

for rows.Next() {
    if err := foo(rs); err != nil {
        // Handle error and return
        return
    }
}
로그인 후 복사

Rows.Close()를 사용한 오류 처리

Rows.Close()가 오류를 반환한다는 점에 유의하는 것이 중요합니다. 이 오류를 처리하려면 Rows.Close()에 대한 지연 호출이 포함된 익명 함수를 사용할 수 있습니다.

func foo(rs *db.Rows) (err error) {
    fields, err := db.Query(...)
    if err != nil {
        return fmt.Errorf("db.Query error: %w", err)
    }
    defer func() {
        if err = fields.Close(); err != nil {
            err = fmt.Errorf("Rows.Close() error: %w", err)
        }
    }()

    // do something with `fields`
    return nil
}
로그인 후 복사

위 내용은 Go에서 리소스를 적절하게 해제하려면 루프에서 `defer` 문을 어디에 배치해야 합니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

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