Форум программистов, компьютерный форум, киберфорум
React/ReactJS
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.67/3: Рейтинг темы: голосов - 3, средняя оценка - 4.67
0 / 0 / 1
Регистрация: 11.05.2021
Сообщений: 92

socket.io Дублирование сообщений в чате ReactJS App

25.10.2022, 12:20. Показов 904. Ответов 9
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Когда я отправляю сообщение, на первом клиенте(отправителе) все в порядке, сообщение не дублируется. Но у остальных пользователей комнаты (получателей) сообщение дублируется. В то же время, если войдет новый пользователь или какой-то пользователь перезайдет в комнату, повторяющихся сообщений не будет (я имею в виду старые сообщения. Новые также будут дублироваться). Пожалуйста, подскажите мне, как это исправить?

server.js

JavaScript
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
const express = require('express');
const useSocket = require('socket.io');
 
const app = express();
const server = require('http').Server(app);
const io = require('socket.io')(server);
 
app.use(express.json());
const rooms = new Map();
 
app.use(function(req, res, next) {
    res.header("Access-Control-Allow-Origin", "http://localhost:3000"); // update to match the domain you will make the request from
    res.header("Access-Control-Allow-Headers", "Origin, X-Requested-With, Content-Type, Accept");
    next();
  });
 
  app.get('/rooms/:id', (req, res) => {
    const { id: roomId } = req.params;
    console.log(roomId);
    const obj = rooms.has(roomId)
      ? {
          users: [...rooms.get(roomId).get('users').values()],
          messages: [...rooms.get(roomId).get('messages').values()],
        }
      : { users: [], messages: [] };
    res.json(obj);
  });
 
  app.post('/rooms', (req, res) => {
    const { roomId, userName } = req.body;
    if (!rooms.has(roomId)) {
      rooms.set(
        roomId,
        new Map([
          ['users', new Map()],
          ['messages', []],
        ]),
      );
    }
    res.send();
  });
  
  io.on('connection', (socket) => {
    socket.on('ROOM:JOIN', ({ roomId, userName }) => {
      socket.join(roomId);
      rooms.get(roomId).get('users').set(socket.id, userName);
      const users = [...rooms.get(roomId).get('users').values()];
      socket.broadcast.to(roomId).emit('ROOM:SET_USERS', users);
    });
  
    socket.on('ROOM:NEW_MESSAGE', ({ roomId, userName, text }) => {
      const obj = {
        userName,
        text,
      };
      rooms.get(roomId).get('messages').push(obj);
      socket.broadcast.to(roomId).emit('ROOM:NEW_MESSAGE', obj);
    });
  
    socket.on('disconnect', () => {
      rooms.forEach((value, roomId) => {
        if (value.get('users').delete(socket.id)) {
          const users = [...value.get('users').values()];
          socket.broadcast.to(roomId).emit('ROOM:SET_USERS', users);
        }
      });
    });
  
    console.log('user connected', socket.id);
  });
  
  server.listen(9999, (err) => {
    if (err) {
      throw Error(err);
    }
    console.log('Server is running!');
  });
App.js

JavaScript
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
import React from 'react';
import axios from 'axios';
import JoinBlock from './components/JoinBlock';
import Chat from './components/Chat';
import socket from './socket'
import reducer from './reducer'
 
 
function App() {
  const [state, dispatch] = React.useReducer(reducer, {
    joined: false,
    roomId: null,
    userName: null,
    users: [],
    messages: [],
  });
 
  const onLogin = async (obj) => {
    dispatch({
      type: 'JOINED',
      payload: obj,
    });
    socket.emit('ROOM:JOIN', obj);
    const { data } = await axios.get(`/rooms/${obj.roomId}`);
    dispatch({
      type: 'SET_DATA',
      payload: data,
    });
  };
 
  const setUsers = (users) => {
    dispatch({
      type: 'SET_USERS',
      payload: users,
    });
  };
 
  const addMessage = (message) => {
    dispatch({
      type: 'NEW_MESSAGE',
      payload: message,
    });
  };
 
  React.useEffect(() => {
    socket.on('ROOM:SET_USERS', setUsers);
    socket.on('ROOM:NEW_MESSAGE', addMessage);
  }, []);
 
  window.socket = socket;
 
  return (
    <div className="wrapper">
      {!state.joined ? (
        <JoinBlock onLogin={onLogin} />
      ) : (
        <Chat {...state} onAddMessage={addMessage} />
      )}
    </div>
  );
}
 
export default App;
Миниатюры
socket.io Дублирование сообщений в чате ReactJS App  
0
Лучшие ответы (1)
Programming
Эксперт
39485 / 9562 / 3019
Регистрация: 12.04.2006
Сообщений: 41,671
Блог
25.10.2022, 12:20
Ответы с готовыми решениями:

Вывод сообщений в чате
Ребзя привет! В общем сообщения в чате записываются в файл, хочу переписать на mysql. Добавить сообщения в таблицу я смог, а вот...

Организовать сохранение сообщений в чате
Организовать сохранение сообщений в чате. Есть сохранение пароля и логина.Бд тоже создана.

Ошибка - не видно сообщений в чате на Js
Писал код чата по видео (на ютубе название &quot;Разработка простого чата Express js / Socket.io за 30 минут&quot;) Столкнулся с ошибкой, что...

9
Молодой техлид)
Эксперт JSЭксперт HTML/CSS
 Аватар для mr_dramm
1818 / 1056 / 329
Регистрация: 17.07.2021
Сообщений: 2,147
Записей в блоге: 14
25.10.2022, 15:23
Лучший ответ Сообщение было отмечено Blakky как решение

Решение

Если react 18 и включен strict mode хук useEffect выполнится дважды

JavaScript
1
2
3
4
  React.useEffect(() => {
    socket.on('ROOM:SET_USERS', setUsers);
    socket.on('ROOM:NEW_MESSAGE', addMessage);
  }, []);
попробуйте так

JavaScript
1
2
3
4
5
6
7
8
9
10
11
12
13
React.useEffect(() => {
  let ignore = false;
  Promise.resolve().then(() => {
    if (!ignore) {
      socket.on("ROOM:SET_USERS", setUsers);
      socket.on("ROOM:NEW_MESSAGE", addMessage);
    }
  });
 
  return () => {
    ignore = true;
  };
}, []);
этот эффект проявляется в функциях с side effect, если работать с локальным состоянием компонента то не нужно думать о двойном вызове
1
0 / 0 / 1
Регистрация: 11.05.2021
Сообщений: 92
25.10.2022, 15:52  [ТС]
mr_dramm, попробовал. Теперь оно повторяется по 3 раза, а не 2.. Может, вы где-нибудь ошиблись?
0
0 / 0 / 1
Регистрация: 11.05.2021
Сообщений: 92
25.10.2022, 15:59  [ТС]
mr_dramm, в addMessage добавил console.log('Сообщение отправлено', message), получил следующее: (картинка)

Как это пофиксить? :c

(кусок reducer.js)
JavaScript
1
2
3
4
5
case 'NEW_MESSAGE':
  return {
    ...state,
    messages: [...state.messages, action.payload],
  };
Миниатюры
socket.io Дублирование сообщений в чате ReactJS App  
0
0 / 0 / 1
Регистрация: 11.05.2021
Сообщений: 92
25.10.2022, 16:18  [ТС]
в reducer, там где NEW_MESSAGE тоже прокинул console.log(). У отправителя 2 повтора, у получателя 4....
0
Молодой техлид)
Эксперт JSЭксперт HTML/CSS
 Аватар для mr_dramm
1818 / 1056 / 329
Регистрация: 17.07.2021
Сообщений: 2,147
Записей в блоге: 14
25.10.2022, 16:26
Я надеюсь вы сделали замену хука useEffect, а не добавили еще кода. Как могло получиться 3 вызова?

Ошибки с ключем пофиксите он итоже косвено могут задваивать сообщение.
Ошибка с установкой соединения уже была или только появилась?

Кажется нужно больше кода анализировать

Добавлено через 3 минуты
Ответьте сначала на вопрос у Вас React 18? или меньше версия? посмотреите в package json

Добавлено через 1 минуту
Если меньше 18 то мои правки просто бесполезны, они не должны были вообще никак повлиять ил можно не использовать
0
0 / 0 / 1
Регистрация: 11.05.2021
Сообщений: 92
25.10.2022, 16:50  [ТС]
Цитата Сообщение от mr_dramm Посмотреть сообщение
Ошибка с установкой соединения уже была или только появилась?
в этот момент просто не был запущен сервер
Цитата Сообщение от mr_dramm Посмотреть сообщение
Я надеюсь вы сделали замену хука useEffect, а не добавили еще кода. Как могло получиться 3 вызова?
не знаю, как получилось три вызова, но я просто заменил свой useeffect вашим.

Добавлено через 1 минуту
Цитата Сообщение от mr_dramm Посмотреть сообщение
Ответьте сначала на вопрос у Вас React 18? или меньше версия? посмотреите в package json
"react": "^18.2.0"

Добавлено через 17 минут
не знаю, вряд ли в chat.jsx проблема..
Code
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
import React from 'react';
import socket from '../socket';
 
function Chat({ users, messages, userName, roomId, onAddMessage }) {
  const [messageValue, setMessageValue] = React.useState('');
  const messagesRef = React.useRef(null);
 
  const onSendMessage = () => {
    socket.emit('ROOM:NEW_MESSAGE', {
      userName,
      roomId,
      text: messageValue,
    });
    onAddMessage({ userName, text: messageValue });
    setMessageValue('');
  };
 
  React.useEffect(() => {
    messagesRef.current.scrollTo(0, 99999);
  }, [messages]);
 
  return (
    <div className="chat">
      <div className="chat-users">
        Room: <b>{roomId}</b>
        <hr />
        <b>Online ({users.length}):</b>
        <ul>
          {users.map((name, index) => (
            <li key={name + index}>{name}</li>
          ))}
        </ul>
      </div>
      <div className="chat-messages">
        <div ref={messagesRef} className="messages">
          {messages.map((message) => (
            <div className="message">
              <p>{message.text}</p>
              <div>
                <span>{message.userName}</span>
              </div>
            </div>
          ))}
        </div>
        <form>
          <textarea
            value={messageValue}
            onChange={(e) => setMessageValue(e.target.value)}
            className="form-control"
            rows="3"></textarea>
          <button onClick={onSendMessage} type="button" className="btn btn-primary">
            Send
          </button>
        </form>
      </div>
    </div>
  );
}
0
Молодой техлид)
Эксперт JSЭксперт HTML/CSS
 Аватар для mr_dramm
1818 / 1056 / 329
Регистрация: 17.07.2021
Сообщений: 2,147
Записей в блоге: 14
25.10.2022, 17:00
добавьте тут key
JavaScript
1
2
3
4
5
6
7
8
          {messages.map((message, i) => (
            <div className="message" key={i}>
              <p>{message.text}</p>
              <div>
                <span>{message.userName}</span>
              </div>
            </div>
          ))}
1
0 / 0 / 1
Регистрация: 11.05.2021
Сообщений: 92
25.10.2022, 17:58  [ТС]
mr_dramm, подправил. теперь той ошибки в консоли нет, но сообщение по прежнему дублируется...

Добавлено через 20 минут
mr_dramm, добавил потом ваш useeffect, и всё заработало! Спасибо огромное
0
Молодой техлид)
Эксперт JSЭксперт HTML/CSS
 Аватар для mr_dramm
1818 / 1056 / 329
Регистрация: 17.07.2021
Сообщений: 2,147
Записей в блоге: 14
25.10.2022, 18:38
Если интересно про двойной запуск useEffect в режимеотладки на react 18 strict mode прочитать

Этот эффект срабатывает только при монтировании компонента т.е. когда у useEffect нет зависимостей

JavaScript
1
useEffect(()=>{....},[тут нет зависимостей])
1
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
inter-admin
Эксперт
29715 / 6470 / 2152
Регистрация: 06.03.2009
Сообщений: 28,500
Блог
25.10.2022, 18:38
Помогаю со студенческими работами здесь

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

Разделение сообщений в чате на группы
Всем привет, у меня тут проблема которую я без понятия как решить. В общем для одного проекта нужен чат, его я сделал, но людишки хотят что...

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

twitch bot не видит сообщений в чате
Здравствуйте. Я новичок в программировании, хотел сделать бота для твича, чтобы он реагировал на команды из чата. Бот подключается к...

Отодвигание listView при написании сообщений в чате
Как отодвинуть listView при написании сообщений в чате? Вот мой xml &lt;?xml version=&quot;1.0&quot; encoding=&quot;utf-8&quot;?&gt; ...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
10
Ответ Создать тему
Новые блоги и статьи
Символьное дифференцирование
igorrr37 13.02.2026
/ * Программа принимает математическое выражение в виде строки и выдаёт его производную в виде строки и вычисляет значение производной при заданном х Логарифм записывается как: (x-2)log(x^2+2) -. . .
Камера Toupcam IUA500KMA
Eddy_Em 12.02.2026
Т. к. у всяких "хикроботов" слишком уж мелкий пиксель, для подсмотра в ESPriF они вообще плохо годятся: уже 14 величину можно рассмотреть еле-еле лишь на экспозициях под 3 секунды (а то и больше),. . .
И ясному Солнцу
zbw 12.02.2026
И ясному Солнцу, и светлой Луне. В мире покоя нет и люди не могут жить в тишине. А жить им немного лет.
«Знание-Сила»
zbw 12.02.2026
«Знание-Сила» «Время-Деньги» «Деньги -Пуля»
SDL3 для Web (WebAssembly): Подключение Box2D v3, физика и отрисовка коллайдеров
8Observer8 12.02.2026
Содержание блога Box2D - это библиотека для 2D физики для анимаций и игр. С её помощью можно определять были ли коллизии между конкретными объектами и вызывать обработчики событий столкновения. . . .
SDL3 для Web (WebAssembly): Загрузка PNG с прозрачным фоном с помощью SDL_LoadPNG (без SDL3_image)
8Observer8 11.02.2026
Содержание блога Библиотека SDL3 содержит встроенные инструменты для базовой работы с изображениями - без использования библиотеки SDL3_image. Пошагово создадим проект для загрузки изображения. . .
SDL3 для Web (WebAssembly): Загрузка PNG с прозрачным фоном с помощью SDL3_image
8Observer8 10.02.2026
Содержание блога Библиотека SDL3_image содержит инструменты для расширенной работы с изображениями. Пошагово создадим проект для загрузки изображения формата PNG с альфа-каналом (с прозрачным. . .
Установка Qt-версии Lazarus IDE в Debian Trixie Xfce
volvo 10.02.2026
В общем, достали меня глюки IDE Лазаруса, собранной с использованием набора виджетов Gtk2 (конкретно: если набирать текст в редакторе и вызвать подсказку через Ctrl+Space, то после закрытия окошка. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru