Назад к вопросам
Яндекс Маркет
Middle
2
Наше приложение-чат должно отображать новые сообщения, которые приходят с сервера, как можно быстрее. Сообщение имеет формат: interface Message { id: number text: string } Id самого первого сообщения = 1, а id каждого следующего сообщения на 1 больше, чем id предыдущего. Нам нужно выводить сообщения в правильном порядке, однако сервер не гарантирует правильный порядок сообщений, отправляемых в наше приложение. Таймлайн: // (приходит) [phone] // (рисуем) . 1 2 3 . . 4 5 6 7 8 Отображать сообщения нужно с помощью функции render: render(message)
Компании, где спрашивали
Ответ от нейросети
sobes.tech AI
Чтобы отображать сообщения в правильном порядке, несмотря на возможный приход сообщений с сервера в произвольном порядке, можно реализовать буферизацию и последовательный рендеринг.
Идея:
- Хранить в памяти последний отрендеренный id сообщения (например,
lastRenderedId). - При получении нового сообщения:
- Если
message.id === lastRenderedId + 1, сразу рендерим и увеличиваемlastRenderedId. - Если
message.id > lastRenderedId + 1, сохраняем сообщение в буфер (например, объект или Map). - После рендера сообщения проверяем буфер, есть ли следующее сообщение (
lastRenderedId + 1), и если есть — рендерим его, обновляемlastRenderedIdи повторяем проверку.
- Если
Пример реализации:
const buffer = new Map();
let lastRenderedId = 0;
function onMessageReceived(message) {
if (message.id === lastRenderedId + 1) {
render(message);
lastRenderedId++;
// Проверяем, есть ли в буфере следующее сообщение
while (buffer.has(lastRenderedId + 1)) {
const nextMsg = buffer.get(lastRenderedId + 1);
buffer.delete(lastRenderedId + 1);
render(nextMsg);
lastRenderedId++;
}
} else if (message.id > lastRenderedId + 1) {
// Сообщение пришло раньше времени, сохраняем в буфер
buffer.set(message.id, message);
}
// Если message.id <= lastRenderedId, это уже отрендеренное или дубликат — игнорируем
}
Такой подход гарантирует, что сообщения будут отображаться строго по порядку, а новые сообщения, пришедшие раньше времени, будут отложены до появления всех предыдущих.