Sobes.tech
Back to questions
Senior
4

/* Вы писатель, ведёте свой мини блог и там публикуете статьи. Пользователи(читатели) вашего блога, после авторизации, могут их просматривать\читать. Необходимо реализовать функциональность подсчета уникальных просмотров за день для каждой статьи. Функциональные требования: - Статистика за сегодня: Автор может видеть, сколько уникальных пользователей просмотрели его статью сегодня (данные накапливаются в реальном времени). - Статистика за прошедшие дни: Автор может видеть статистику за любой прошедший день (за вчера). Технические ограничения и условия: - Приложение работает на 1 сервере(1 pod) - Хранилище в памяти(in memory) - Время по UTC0 */

Companies where asked
А7

Answer from AI

sobes.tech AI

Для подсчёта уникальных просмотров статей за день с хранением данных в памяти и одним сервером можно использовать следующую структуру:

  • Для каждого дня (по дате UTC) хранить мапу: map[articleID]map[userID]bool, где ключи — ID статьи и ID пользователя.
  • При просмотре статьи авторизованным пользователем добавлять userID в мапу для соответствующей статьи и текущей даты.
  • Для получения статистики за сегодня или прошедший день считать длину внутренней мапы пользователей для статьи.

Пример на Go:

type Stats struct {
    mu sync.RWMutex
    data map[string]map[string]map[string]bool // date -> articleID -> userID -> bool
}

func NewStats() *Stats {
    return &Stats{data: make(map[string]map[string]map[string]bool)}
}

func (s *Stats) AddView(articleID, userID string) {
    s.mu.Lock()
    defer s.mu.Unlock()
    date := time.Now().UTC().Format("2006-01-02")
    if s.data[date] == nil {
        s.data[date] = make(map[string]map[string]bool)
    }
    if s.data[date][articleID] == nil {
        s.data[date][articleID] = make(map[string]bool)
    }
    s.data[date][articleID][userID] = true
}

func (s *Stats) GetUniqueViews(articleID, date string) int {
    s.mu.RLock()
    defer s.mu.RUnlock()
    if s.data[date] == nil || s.data[date][articleID] == nil {
        return 0
    }
    return len(s.data[date][articleID])
}

Такой подход позволяет в реальном времени накапливать уникальные просмотры и получать статистику за любой день. При необходимости можно добавить периодическую очистку старых данных, чтобы не накапливать память.