- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
在加入之前,我学习了一些关于如何显示在聊天室中发送的消息的教程,但我不知道如何在 React 中显示它们,并且我在服务器端有几个问题。
客户端,在构造函数中:
this.state = {
msg: "",
messages: []
};
客户端,我有一个表单,单击按钮将通过此函数将消息发送到服务器:
sendMessage(e) {
e.preventDefault();
let msg = this.state.msg;
this.socket.emit("sendMessage", msg);
this.setState({ msg: "" });
}
服务器端,我有一个消息的 Mongoose 模式,名为 Message ,数据库中的集合是消息。
const Message = new mongoose.Schema({
sender: {
type: "string"
},
message: {
type: "string"
}
});
var messages = [];
io.on("connection", (socket, user) => {
var user = socket.request.session.user;
Message.find({}).exec((err, messages) => {
if (err) throw err;
console.log(messages);
io.emit("showingPastMessages", messages);
});
console.log(messages) 在 PowerShell 中显示 Mongo 中保存在 JavaScript 对象数组中的所有消息(条目)?[{id_:4qxxx,发件人:'user123',消息:'你好!'},{id_:5exxx,发件人:'user456',消息:'嗨!'}]我想知道是否可以仅访问发送者和消息属性以将其发送到客户端?像 messages.sender + messages.message 这样的东西,因为当我 console.log(messages.message) 它显示未定义
这里是服务器接收发送的消息并将其保存在 Mongo 中的位置。
socket.on("sendMessage", function(msg) {
var newMsg = new Message({ message: msg, sender: user });
newMsg.save(function(err) {
if (err) {
console.log(err);
} else {
messages.push(newMsg);
console.log(newMsg);
console.log(messages);
}
});
});
console.log(newMsg) 显示最新发送的消息,但 console.log(messages) 不显示之前的消息,只显示最新的消息,为什么?
那么在 React 中,我应该在构造函数中的 ComponentDidMount() 中有类似的东西?如果应该与 prevState 一起使用
this.socket.on("showingPastMessages", function(messages){
this.setState({ ...this.state.messages, messages})
});
你能给我一些建议吗?
这是我用于检索数据的客户端代码:
class Chat extends Component {
constructor(props) {
super(props);
this.state = {
msg: "",
messages: []
};
var socket = io('http://localhost:3000');
this.socket.on("history", function(messages) {
console.log(messages);
});
}
最佳答案
这是一个好的开始。我建议做的是让这变得更加事件驱动。为此,您需要在连接时将 socket 添加到房间。有多少房间以及如何分割房间将取决于您的应用程序,但我将演示基本想法。
首先,当套接字连接到您的服务器时,将该套接字添加到房间并立即发出您的聊天历史记录。
io.on('connection', async (socket) => {
socket.user = socket.request.session.user;
socket.join('chat');
try {
const messages = await Message.find({});
socket.emit('history', messages);
} catch (err) {
// Handle this error properly.
console.error(err);
}
});
然后,当您收到消息时,您将需要保存该消息并将其发送到聊天室中的所有套接字。
socket.on("sendMessage", (msg, callback) => {
const message = new Message({ message: msg, sender: socket.user });
message.save(function(err) {
if (err) {
return callback(err);
}
io.to('chat').emit('message', message);
});
});
最后,在客户端,您需要监听历史事件。当您收到它时,您需要清除当前的聊天记录并将其替换为服务器告诉您的内容。也许这看起来像
socket.on('history', (messages) => {
this.setState({ messages });
});
您还需要监听此 message
事件,但通过此事件,您只需将该消息附加到您的历史记录中。这可能看起来像
socket.on('message', (message) => {
this.setState({ messages: [ ...messages, message ] });
});
警告一句,如果您告诉服务器有一条新消息,请不要将其添加到您的 messages
状态数组中,直到收到 message
事件。如果这样做,您会注意到双重消息。例如,这可能看起来像
onSendMessage(evnt) {
evnt.preventDefault();
socket.emit("sendMessage", msg);
this.setState({ msg: "" });
}
注意:收到 OP 的一些反馈后,我想添加一个部分,说明将事件处理程序附加到套接字的位置(即所有 socket.on
s)。我的建议是在 io.on('connection')
回调中的文件底部定义 Message
架构的文件中添加此代码。例如,
const Message = new mongoose.Schema({/*...*/});
io.on('connection', (socket) => {
// Everything else I wrote above...
socket.on('sendMessage', (msg, callback) => {
// ...
});
});
在客户端,事件处理程序可能会在安装聊天组件时注册。例如,
class ChatComponent extends React.Component {
componentDidMount() {
this.socket = io('https://your-server-or-localhost/');
this.socket.on('history', (messages) => {
// See above...
});
}
}
关于node.js - Socket.io - React : How to retrieve saved socket. io 数据(MongoDB)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54897027/
IO 设备如何知道属于它的内存中的值在memory mapped IO 中发生了变化? ? 例如,假设内存地址 0 专用于保存 VGA 设备的背景颜色。当我们更改 memory[0] 中的值时,VGA
我目前正在开发一个使用Facebook sdk登录(通过FBLoginView)的iOS应用。 一切正常,除了那些拥有较旧版本的facebook的人。 当他们按下“使用Facebook登录”按钮时,他
假设我有: this - is an - example - with some - dashesNSRange将使用`rangeOfString:@“-”拾取“-”的第一个实例,但是如果我只想要最后
Card.io SDK提供以下详细信息: 卡号,有效期,月份,年份,CVV和邮政编码。 如何从此SDK获取国家名称。 - (void)userDidProvideCreditCardInfo:(Car
iOS 应用程序如何从网络服务下载图片并在安装过程中将它们安装到用户的 iOS 设备上?可能吗? 最佳答案 您无法控制应用在用户设备上的安装,因此无法在安装过程中下载其他数据。 只需在安装后首次启动应
我曾经开发过一款企业版 iOS 产品,我们公司曾将其出售给大型企业,供他们的员工使用。 该应用程序通过 AppStore 提供,企业用户获得了公司特定的配置文件(包含应用程序配置文件)以启用他们有权使
我正在尝试将 Card.io SDK 集成到我的 iOS 应用程序中。我想为 CardIO ui 做一个简单的本地化,如更改取消按钮标题或“在此保留信用卡”提示文本。 我在 github 上找到了这个
我正在使用 CardIOView 和 CardIOViewDelegate 类,没有可以设置为 YES 的 BOOL 来扫描 collectCardholderName。我可以看到它在 CardIOP
我有一个集成了通话工具包的 voip 应用程序。每次我从我的 voip 应用程序调用时,都会在 native 电话应用程序中创建一个新的最近通话记录。我在 voip 应用程序中也有自定义联系人(电话应
iOS 应用程序如何知道应用程序打开时屏幕上是否已经有键盘?应用程序运行后,它可以接收键盘显示/隐藏通知。但是,如果应用程序在分屏模式下作为辅助应用程序打开,而主应用程序已经显示键盘,则辅助应用程序不
我在模拟器中收到以下错误: ImageIO: CGImageReadSessionGetCachedImageBlockData *** CGImageReadSessionGetCachedIm
如 Apple 文档所示,可以通过 EAAccessory Framework 与经过认证的配件(由 Apple 认证)进行通信。但是我有点困惑,因为一些帖子告诉我它也可以通过 CoreBluetoo
尽管现在的调试器已经很不错了,但有时找出应用程序中正在发生的事情的最好方法仍然是古老的 NSLog。当您连接到计算机时,这样做很容易; Xcode 会帮助弹出日志查看器面板,然后就可以了。当您不在办公
在我的 iOS 应用程序中,我定义了一些兴趣点。其中一些有一个 Kontakt.io 信标的名称,它绑定(bind)到一个特定的 PoI(我的意思是通常贴在信标标签上的名称)。现在我想在附近发现信标,
我正在为警报提示创建一个 trigger.io 插件。尝试从警报提示返回数据。这是我的代码: // Prompt + (void)show_prompt:(ForgeTask*)task{
您好,我是 Apple iOS 的新手。我阅读并搜索了很多关于推送通知的文章,但我没有发现任何关于 APNS 从 io4 到 ios 6 的新更新的信息。任何人都可以向我提供 APNS 如何在 ios
UITabBar 的高度似乎在 iOS 7 和 8/9/10/11 之间发生了变化。我发布这个问题是为了让其他人轻松找到答案。 那么:在 iPhone 和 iPad 上的 iOS 8/9/10/11
我想我可以针对不同的 iOS 版本使用不同的 Storyboard。 由于 UI 的差异,我将创建下一个 Storyboard: Main_iPhone.storyboard Main_iPad.st
我正在写一些东西,我将使用设备的 iTunes 库中的一部分音轨来覆盖 2 个视频的组合,例如: AVMutableComposition* mixComposition = [[AVMutableC
我创建了一个简单的 iOS 程序,可以顺利编译并在 iPad 模拟器上运行良好。当我告诉 XCode 4 使用我连接的 iPad 设备时,无法编译相同的程序。问题似乎是当我尝试使用附加的 iPad 时
我是一名优秀的程序员,十分优秀!