// Написать имплементации кеша с ttl (time to live) для структуры User. // В качестве ключа можно использовать User.ID // мы хотим чтобы было потокобезопасно // мы не хотим получать записи с просроченным ttl при операции чтения // мы хотим чтобы записи с просроченным ttl удалялись и не хранились // API get/set/del
Ответ от нейросети
sobes.tech AI
go package main
import ( "sync" "time" )
type User struct { ID string // другие поля }
type cacheItem struct { user User expiresAt time.Time }
type UserCache struct { mu sync.RWMutex items map[string]cacheItem }
func NewUserCache() *UserCache { c := &UserCache{ items: make(map[string]cacheItem), } // Запускаем горутину для периодической очистки просроченных записей go func() { for { time.Sleep(time.Minute) // интервал очистки c.cleanup() } }() return c }
func (c *UserCache) Set(user User, ttl time.Duration) { c.mu.Lock() defer c.mu.Unlock() c.items[user.ID] = cacheItem{ user: user, expiresAt: time.Now().Add(ttl), } }
func (c *UserCache) Get(id string) (User, bool) { c.mu.RLock() item, found := c.items[id] c.mu.RUnlock() if !found { return User{}, false } if time.Now().After(item.expiresAt) { // Запись просрочена, удаляем c.mu.Lock() delete(c.items, id) c.mu.Unlock() return User{}, false } return item.user, true }
func (c *UserCache) Del(id string) { c.mu.Lock() defer c.mu.Unlock() delete(c.items, id) }
func (c *UserCache) cleanup() { c.mu.Lock() defer c.mu.Unlock() now := time.Now() for id, item := range c.items { if now.After(item.expiresAt) { delete(c.items, id) } } }
// Пример использования // func main() { // cache := NewUserCache() // user := User{ID: "123"} // cache.Set(user, 5*time.Minute) // u, ok := cache.Get("123") // if ok { // fmt.Println("User found:", u) // } // }