node.js - 多个监听器和removeListener删除所有Socket.io

标签 node.js reactjs express sockets socket.io

伙计们!我正在写一个 socket.io 聊天。当用户发送消息时,socket.on 事件发生。一切正常,但听众的数量呈指数级增长。我尝试使用 socket.off/socket.remove... 删除监听器,从连接中单独取出事件 socket.on - 没有任何效果。拜托,请帮我弄清楚。我正在使用 React、Node、socket 和 mongoDB

服务器部分:

const express = require("express");
const app = express();
const cors = require("cors");
const { UserModel } = require("./models");

app.use(cors());

app.use(express.json());
app.use(express.urlencoded({ extended: true }));

app.options("http://localhost:3000", cors());
const mongoose = require("mongoose");
require("dotenv").config();

const http = require("http").createServer(app);
const io = require("socket.io")(http, {
  cors: {
    origin: ["http://localhost:3002"],
  },
});

http.listen(3001, function () {
  console.log("Server is running");
});

io.on("connection", (socket) => {
  console.log("Socket connected", socket.id);

  socket.on("message:send", (data) => {
    console.log("Пришло в socket.onMessage", data);
    socket.emit("message:fromServer", data);
    // socket.removeListener("message:fromServer");
  });
});

const { userApi } = require("./api/userApi");
app.use("/", userApi);

app.use((err, _, res, __) => {
  return res.status(500).json({
    status: "fail",
    code: 500,
    message: err.message,
  });
});

const { PORT, DB_HOST } = process.env;

const dbConnection = mongoose.connect(DB_HOST, {
  useNewUrlParser: true,
  useFindAndModify: false,
  useCreateIndex: true,
  useUnifiedTopology: true,
});

dbConnection
  .then(() => {
    console.log("DB connect");
    app.listen(PORT || 3000, () => {
      console.log("server running");
    });
  })
  .catch((err) => {
    console.log(err);
  });

客户端部分: io.js

import { io } from "socket.io-client";

export const socket = io("http://localhost:3001/");

消息组件

import React from "react";
import { useState } from "react";
// import { useSelector } from "react-redux";
// import { getMessages } from "../../Redux/selectors";
import { socket } from "../helpers/io";
import Message from "../Message/Message";
import { nanoid } from "nanoid";

export default function MessageFlow() {
  const [message, setMessage] = useState([]);

  socket.on("message:fromServer", (data) => {
    console.log("На фронт пришло сообщение: ", data);
    setMessage([...message, data]);
    // setMessage((message) => [...message, data]);
    console.log("Массив сообщений компонента MessageFlow", message);
    console.log(socket.io.engine.transports);
    // socket.off();
    // getEventListeners(socket)['testComplete'][0].remove()
    // socket.removeListener("message:fromServer");
  });

  //   const dispatch = useDispatch();
  // const allMessages = useSelector(getMessages);
  return (
    <div id="mainDiv">
      {message &&
        message.map((i) => {
          // return <Message />;
          return <Message content={i.userId} id={nanoid()} />;
        })}
    </div>
  );
}

消息表单 - 发送过程的开始

import React, { useState } from "react";
import { useDispatch, useSelector } from "react-redux";
import { sendMessage } from "../../Redux/Chat/Chat-operations";
import { getUser } from "../../Redux/selectors";
import { getToken } from "../../Redux/Auth/Auth-selectors";
import { socket } from "../helpers/io";
import { useEffect } from "react";
import styles from "./MessageForm.module.css";

export default function MessageForm() {
  const [message, setMessage] = useState("");
  const dispatch = useDispatch();
  const userId = useSelector(getUser);
  const currentToken = useSelector(getToken);
  // const getAll = useSelector(allContacts);

  const updateMessage = (evt) => {
    setMessage(evt.target.value);
  };

  const handleSubmit = (e) => {
    e.preventDefault();
    if (message) {
      socket.emit("message:send", { userId: message });
      dispatch(
        sendMessage(
          {
            _id: userId,
            text: message,
          },
          currentToken
        )
      );
    } else {
      alert(`Молчать будем?`);
    }
  };
  return (
    <div className={styles.messageInputContainer}>
      <form>
        <input
          type="text"
          value={message}
          onChange={updateMessage}
          required
          className={styles.messageInput}
          placeholder="Type message to send"
        />
        <button
          type="submit"
          className={styles.messageAddBtn}
          onClick={handleSubmit}
        >
          Send
        </button>
      </form>
    </div>
  );
}

最佳答案

你在每个渲染上添加一个监听器,你应该使用useEffect钩子(Hook)

export default function MessageFlow() {

  useEffect(()=>{ // triggered on component mount or when dependency array change
      const callback = (data) => {
        // what you want to do
      }
      socket.on("message:fromServer", callback);
      return () => { // on unmount, clean your listeners
        socket.removeListener('message:fromServer', callback);
      }
  }, []) // dependency array : list variables used in your listener
  // [...]
}

关于node.js - 多个监听器和removeListener删除所有Socket.io,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/68328371/

相关文章:

javascript - React 中 this.setState 的异步特性

javascript - 如何在开发中运行多个 Node.js 应用程序

javascript - NodeJS 分配变量不起作用

json - 基于 Websocket 的应用程序,安全性和实时性良好实践?

node.js - 在 Linux Jenkins CI 上使用 Windows 中的 node_modules...好还是坏?

reactjs - React Redux 在 reducer 中分配箭头功能警告

node.js - 使用 Sequelize/Node.js,如何将笨拙的 TEXT 数据类型插入到 postgres 数据库中?

node.js - 我如何一起使用passportjs和reactjs?

node.js - 启动npm : npm ERR!代码ELIFECYCLE时遇到此问题

reactjs - 无法在 react native 应用程序中使用 react 导航选项卡,从而出现找不到模块的问题?