> 백엔드 개발 > Golang > 업데이트와 오류를 적절하게 처리하는 구성 캐시를 설계하는 방법은 무엇입니까?

업데이트와 오류를 적절하게 처리하는 구성 캐시를 설계하는 방법은 무엇입니까?

Mary-Kate Olsen
풀어 주다: 2024-10-25 06:43:02
원래의
840명이 탐색했습니다.

How to Design a Configuration Cache That Handles Updates and Errors Gracefully?

파일에서 구성 로드 및 새 업데이트로 새로 고침

문제 설명:

코드 설계에는 시작 시 파일에서 구성을 로드하고 정기적으로 최신 버전으로 새로 고치는 작업이 포함됩니다. 목표는 다음 요구 사항을 처리하는 메커니즘을 갖추는 것입니다.

  • 구성에 대한 동시 액세스
  • 감지된 구성 변경 사항 다시 로드
  • 최신 구성에 대한 액세스 가능성
  • 업데이트 중 최신 구성에 즉시 액세스
  • 업데이트 실패 시 이전 구성 보존

초기 설계에서는 동시 맵을 활용하여 구성을 저장합니다. , 그러나 업데이트 중 오류로 인해 빈 지도가 발생할 수 있는 문제에 직면합니다.

해결책:

모든 요구 사항을 충족하는 단순화된 설계가 제안됩니다.

CustomerConfig 구조:

캐시할 구성 정의:

type CustomerConfig struct {
    Data map[string]bool
    LoadedAt time.Time
}
로그인 후 복사

loadConfig 함수:

로드 파일의 구성:

func loadConfig() (*CustomerConfig, error) {
    cfg := &CustomerConfig{
        Data:     map[string]bool{},
        LoadedAt: time.Now(),
    }

    // Logic to load files and populate cfg.Data
    // If an error occurs, return it

    // If loading succeeds, return the config
    return cfg, nil
}
로그인 후 복사

ConfigCache 구조:

구성 캐싱 관리:

type ConfigCache struct {
    configMu sync.RWMutex
    config   *CustomerConfig
    closeCh  chan struct{}
}
로그인 후 복사

NewConfigCache 함수:

새 구성 캐시 생성:

func NewConfigCache() (*ConfigCache, error) {
    cfg, err := loadConfig()
    if err != nil {
        return nil, fmt.Errorf("loading initial config failed: %w", err)
    }

    cc := &ConfigCache{
        config:  cfg,
        closeCh: make(chan struct{}),
    }

    // Launch a goroutine to periodically check for changes and load new configs
    go cc.refresher()

    return cc, nil
}
로그인 후 복사

refresher 기능:

정기적으로 구성 변경 사항을 확인하고 캐시를 업데이트합니다.

func (cc *ConfigCache) refresher() {
    ticker := time.NewTicker(1 * time.Minute) // Every minute
    defer ticker.Stop()

    for {
        select {
        case <-ticker.C:
            // Check for changes
            changes := false // Logic to detect changes
            if !changes {
                continue // No changes, continue
            }

            // Changes! Load new config:
            cfg, err := loadConfig()
            if err != nil {
                log.Printf("Failed to load config: %v", err)
                continue // Keep the previous config
            }

            // Apply / store new config
            cc.configMu.Lock()
            cc.config = cfg
            cc.configMu.Unlock()

        case <-cc.closeCh:
            return
        }
    }
}
로그인 후 복사

중지 기능:

새로 고침 고루틴 중지:

func (cc *ConfigCache) Stop() {
    close(cc.closeCh)
}
로그인 후 복사

GetConfig 함수:

현재 구성에 액세스:

func (cc *ConfigCache) GetConfig() *CustomerConfig {
    cc.configMu.RLock()
    defer cc.configMu.RUnlock()
    return cc.config
}
로그인 후 복사

사용:

cc, err := NewConfigCache()
if err != nil {
    // Handle the error appropriately
}

// Access the configuration whenever needed:
cfg := cc.GetConfig()
// Use the configuration here

// Stop the cache refreshing when necessary:
cc.Stop()
로그인 후 복사

이 솔루션은 다음을 보장합니다.

  • 구성에 대한 동시 액세스
  • 변경 사항 감지 시 다시 로드
  • 최신 구성에 대한 접근성
  • 업데이트 중 즉시 접근성
  • 업데이트 실패 시 이전 구성 보존

위 내용은 업데이트와 오류를 적절하게 처리하는 구성 캐시를 설계하는 방법은 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

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