- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我目前正在使用非阻塞 SocketChannel (Java 1.6) 作为 Redis 服务器的客户端。 Redis 直接通过套接字接受纯文本命令,由 CRLF 终止并以类似方式响应,一个简单的例子:
SEND: 'PING\r\n'
RECV: '+PONG\r\n'
Redis 还可以返回大量回复(取决于您的请求),其中包含许多以\r\n 结尾的数据部分,所有这些都作为单个响应的一部分。
我正在使用标准的 while(socket.read() > 0) {//append bytes} 循环从套接字中读取字节并将它们重新组装到客户端回复中。
注意:我没有使用选择器,只是连接到服务器的多个客户端 SocketChannels,等待服务发送/接收命令。
我比较困惑的是SocketChannel.read()方法在非阻塞模式下的contract,具体来说就是如何知道服务器什么时候完成发送,我有完整的消息。
我有几种方法可以防止返回得太快并让服务器有机会回复,但我坚持的一件事是:
基本上,如果我收到至少 1 个字节并且最终 read() 返回 0 那么我知道我已经完成,或者是否有可能,我是否可以相信服务器已完成对我的响应服务器正忙,如果我等待并继续尝试,可能会返回更多字节?
如果即使在 read() 返回 0 字节后(在先前成功读取之后)它可以继续发送字节,那么我不知道如何判断服务器何时完成与我的对话以及-事实上我很困惑 java.io.* 风格的通信甚至会知道服务器何时“完成”。
正如你们所知,读取永远不会返回 -1,除非连接已断开,并且这些是标准的长生命周期数据库连接,因此我不会在每个请求时关闭和打开它们。
我知道一个流行的回应(至少对于这些 NIO 问题)是查看 Grizzly、MINA 或 Netty——如果可能的话,我真的很想在采用一些第 3 方依赖项之前了解这一切在原始状态下是如何工作的.
谢谢。
奖励问题:
我最初认为阻塞 SocketChannel 是解决这个问题的方法,因为我真的不希望调用者做任何事情,直到我处理他们的命令并给他们回复。
如果这最终是一个更好的方法,我有点困惑,只要没有足够的字节来填充给定的缓冲区,就会看到 SocketChannel.read() 阻塞......缺少读取所有字节 -按字节我无法弄清楚这种默认行为实际上是如何使用的...我永远不知道从服务器返回的回复的确切大小,所以我调用了 SocketChannel。 read() 总是阻塞直到超时(此时我终于看到内容位于缓冲区中)。
我不太清楚使用阻塞方法的正确方法,因为它总是在读取时挂起。
最佳答案
查看您的 Redis 规范以获得此答案。
调用 .read()
在一次调用中返回 0 个字节,在后续调用中返回 1 个或多个字节并不违反规则。这是完全合法的。如果有任何事情导致交付延迟,无论是由于网络延迟还是 Redis 服务器缓慢,都可能会发生这种情况。
您寻求的答案与以下问题的答案相同:“如果我手动连接到 Redis 服务器并发送命令,我怎么知道它何时完成向我发送响应以便我可以发送另一个命令? "
答案必须在 Redis 规范中找到。如果服务器在执行完您的命令后没有发送全局 token ,那么这可以在逐个命令的基础上实现。如果 Redis 规范不允许这样做,那么这是 Redis 规范中的错误。他们应该告诉您如何判断他们何时发送了所有数据。这就是 shell 具有命令提示符的原因。 Redis 应该有一个等价物。
如果 Redis 在其规范中没有此功能,那么我建议加入某种计时器功能。对处理套接字的线程进行编码,以在指定时间段(如五秒)内未收到任何数据后发出命令已完成的信号。选择一个明显长于在服务器上执行的最长命令所需的时间。
关于Java NIO : How to know when SocketChannel read() is complete with non-blocking I/O,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4926451/
我正在使用 gmock 并模拟了一个函数 boost::beast::http::response_parser作为输出参数。功能签名看起来像: error_code readFromSocket(b
我的任务是打印由“非元音、元音、非元音”组成的单词列表,即 bab、bac、bad、bad ... 到 zuz。 我已经设法创建了一个代码,它执行前两个字母,但在最后一个循环中丢失并只打印'}' -
我正在尝试使用 label2rgb 生成 RGB 标签切片并使用它来更新 RGB 体积,如下所示: labelRGB_slice=label2rgb(handles.label(:,:,han
我有一个很奇怪的问题。我在 dll 中定义了一个接口(interface),如下所示: public interface IKreator2 { string Name { get; set;
在我的 openshift Drupal 托管中,网络都在 SSL 下 http://domain.com -> https://www.domain.com 确定 http://www.domain
我收到警告“退出构造函数时不可为空的事件‘SomeEvent’必须包含非空值。考虑将事件声明为可空。” 这是我的代码的一个非常简化的版本,它复制了完全相同的问题。我在这里错过了什么?这与 .Net 6
在一次大学考试中,我被要求测试一些 apache 簿记员类/方法,在这样做的过程中,我想在我的参数化测试中使用 mockito。没有 mockito 的测试工作正常但是当我尝试模拟接口(interfa
假设 A 列在 7 行中有以下值: 2 [空白的] 0 -0.3 0 [空白的] 0 如何获取范围(7 行)中非空/空白且不为零的最后一个值?因此,在这种情况下,正确答案是 -0.3。 最佳答案 =I
考虑以下受 this talk 启发的代码: template struct even_common_type_helper_impl; template struct even_common_typ
考虑这段代码, struct A {}; struct B { B(const A&) {} }; void f(B) { cout << "f()"<
考虑下面的类(class)。如果我对它运行 Findbugs,它会在第 5 行但不在第 7 行给我一个错误(“可序列化类中的非 transient 非可序列化实例字段”)。 1 public clas
我正在编写一个 python 脚本来计算 数据包丢失 通过使用 ping IP 地址linux 中的 subprocess 模块。 CSV 文件中保存了多个 IP 地址。当只给出可 ping 目的地时
我只是做文本更改,在文本之前它工作正常。请任何人都可以帮助我。 提前致谢 最佳答案 我已经解决了: ionic cordova 插件rmcordova-plugin-ionic-webview ion
我如何定义 在 persistence.xml 中? 我的项目在 Tomcat 6 和 Tomcat 7 中运行良好。 现在我正在使用 Struts 2 Spring 3.0.5 JPA 2 Jbos
我有一个 maven 仓库中不存在的第三方 jar,我们称它为“a.jar”,它也依赖于至少 20 多个第三方 jar,其中大部分不在 maven 中或者,我们称它们为“b.jar、c.jar、d.j
我已经浏览了各种线程很多小时(不夸张),但一直无法找到一种解决方案组合,使我能够将非 www 和 http 转发到 www 和 https,同时仍然能够查看 php 文件没有扩展名。如下是我的ngin
Scott Meyer 关于非成员函数增加封装并允许更优雅的设计(设计方面)的论点对我来说似乎非常有效。看这里:Article 但是我对此有疑问。 (似乎还有其他人,尤其是库开发人员,他们通常完全忽略
在对类设计的一些事实感到困惑时,特别是函数是否应该是成员,我查看了 Effective c++ 并找到了第 23 条,即 Prefer non-member non-friend functions
我正在尝试使用 firebase 云功能将通知发送到一个点半径的圆内的设备。我能够获取圈内设备的 ID,但无法获取 token ,使用 console.log(token) 打印时 token 为空。
我在我的项目中使用 React-ckeditor 5 包。我得到一个反序列化的 html 数据,我正在使用 React-html-parser 包将它解析成 html 模板,并将这个解析的数据传递给
我是一名优秀的程序员,十分优秀!