- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
在过去的 5 天里,我一直被困在这个问题上,我不知道如何继续。
概述:
我有一个客户端 UI
,它与一个数据处理程序
库交互,数据处理程序
库使用一个网络管理器
库,这是我的问题所在。
更多信息
首先,QT 为 QTcpServer
( Fortune Server ) 和 QTcpSocket
( Fortune Client ) 之间的交互提供了一个基本示例。
因此,我将这段代码实现为我自己的一个极其基本的示例,它运行起来非常棒,没有任何问题。
快速解释:
服务器应用程序运行,单击启动服务器
,然后在客户端,在字段中输入文本并单击连接到服务器
,然后显示文本,简单!
问题:
在我的 network manager
库中执行上面的代码,不会触发 QTcpSocket::readyRead()
在上面的服务器应用程序中。
它连接到服务器
,其中QTcpServer::newConnection()
正如预期的那样,在 client
写入套接字之后立即被触发,但服务器套接字上的 readyRead()
不会触发,但是在给出的示例中会触发。
注意:在这个server-client
应用示例和我现在的应用中使用了相同的端口
和ip地址
,并且服务器也在运行。
更多信息:
上面的代码,我是直接从客户端复制过来的。只有两件事被改变/修改:
这被复制到我的 network mannager::write()
方法中。运行我的应用程序时,QMainWindow
的实例通过 data handler
类传递,并创建我的 network manager
类的实例,它继承了 QObject
并实现了 Q_OBJECT
宏。
代码示例:
//client_UI
类(片段):
data_mananger *dman = new data_mananger(this); //this -> QMainWindow
ReturnObject r = dman->NET_AuthenticateUser_GetToken(Query);
//data_manager
库(片段)
data_mananger::data_mananger(QObject *_parent) :
parent(_parent)
{}
ReturnObject data_mananger::NET_AuthenticateUser_GetToken(QString Query){
//Query like "AUTH;U=xyz@a;P=1234"
//convert query string to char
QByteArray ba = Query.toLatin1();
//send query and get QList return
ReturnCode rCode = networkManager.write(ba);
//...
}
//netman
库(片段)
//.h
class NETMANSHARED_EXPORT netman : public QObject
{
Q_OBJECT
public
netman();
netman(QObject *_parent);
//...
private:
QTcpSocket *tcp_con;
//...
};
//cpp
netman::netman(QObject *_parent) :
parent(_parent)
{
tcp_con = new QTcpSocket(parent);
}
return;
}
serverIP.setAddress(serverInfo.addresses().first().toIPv4Address());
}
ReturnCode netman::write(QByteArray message, int portNumber){
tcp_con->connectToHost(QHostAddress("127.0.0.1"), 5000);
if (!tcp_con->waitForConnected())
{
qDebug(log_lib_netman_err) << "Unable to connect to server";
return ReturnCode::FailedConnecting;
}
if (!tcp_con->isValid()) {
qDebug(log_lib_netman_err) << "tcp socket invalid";
return ReturnCode::SocketError;
}
if (!tcp_con->isOpen()) {
qDebug(log_lib_netman_err) << "tcp socket not open";
return ReturnCode::SocketError;
}
// QByteArray block(message);
QByteArray block;
QDataStream out(&block,QIODevice::WriteOnly);
out.setVersion(QDataStream::Qt_4_0);
out << QString("Hello world");
if (!tcp_con->write(block)){
qDebug(log_lib_netman_err) << "Unable to send data to server";
return ReturnCode::WriteFailed;
}
else{
qDebug(log_lib_netman_info) << "Data block sent";
return ReturnCode::SentSuccess;
}
}
结论:
客户端的核心代码已经完全实现了,但是我看不出为什么会出现这个错误。
非常感谢帮助/建议!
最佳答案
在写入函数的末尾添加一个 tcp_con->flush()
语句。
您没有在接收器中收到 readyRead
信号,因为写入的数据正在缓冲到套接字中,但实际上并未“通过线路”传输。 flush()
命令导致缓冲区被传输。来自 the docs
This function writes as much as possible from the internal write buffer to the underlying network socket, without blocking. If any data was written, this function returns true; otherwise false is returned.
就我而言,在串行端口和刷新方面有很多经验/挫败感。这相当于“你重启了吗?”在套接字调试工具箱中。
如果其他一切正常,您可能不必flush
,但它是特定于应用程序的,取决于套接字的生命周期、TCP 窗口大小、套接字选项设置以及各种其他因素。也就是说,我总是刷新,因为我喜欢完全控制我的套接字,并且我想确保数据在我想要的时候传输。我不认为这是 hack,但在某些情况下,它可能表明存在其他问题。同样,特定于应用程序。
我很确定在 fortune 服务器示例中不需要刷新,因为它们在 sendFortune()
函数的末尾 disconnectFromHost
,并且来自 Qt documentation :
Attempts to close the socket. If there is pending data waiting to be written, QAbstractSocket will enter ClosingState and wait until all data has been written.
如果套接字也被销毁,它也会断开连接,但从我看到的你的代码来看,你也没有这样做,而且缓冲区还没有满,所以可能实际上没有任何东西刺激缓冲区自行刷新.
其他原因可能是:
while(dataToTransmit)
),但实际上条件永远不会变为 false,这会导致事件循环被阻塞。 QAbstractSocket::LowDelayOption
来禁用它,但它可能会对您的吞吐量产生不利影响……它通常用于对延迟敏感的应用程序。关于c++ - 写入 QTcpSocket 并不总是在相反的 QTcpSocket 上发出 readyRead 信号,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42074728/
我不明白 int 63823 为何比 double 1.0 占用更少的空间。在这个特定实例中,int 中是否没有存储更多信息? 最佳答案 I don't understand how an int 6
这可能不是一个直接的代码问题,但它是一个经常出现在 SO 上的问题,我发现阅读它非常有用。 App Store - Help answering “Missing Compliance” (using
我在我们的应用程序中使用 syncfusion 寻呼机和下拉列表请打开以下链接。 https://stackblitz.com/edit/angular-nv6myv?file=src%2Fapp%2
以便解释指针和引用in this question我写了这段代码。 MyClass& MyClass::MyInstance() { static MyClass & myLoca
在 C 和 C++ 中,assert 是一个非常 重量级例程,将错误写入 stdout 并终止程序。在我们的应用程序中,我们实现了一个更强大的 assert 替代品,并为其提供了自己的宏。已尽一切努力
我已经创建了一个 MVC webApi 项目,现在我想使用身份验证和授权。我想我已经实现了这种安全措施,但由于某种原因,有些事情变糟了,当我编写我的凭据并尝试调用一些 webApi 方法时,显示消息“
我发现自己使用一种奇怪的方式向我的函数添加回调函数,我想知道是否有更通用的方式向函数添加回调函数,最好的情况是我的所有函数都检查最后给定的作为函数的参数,如果是,则将其用作回调。 我以前是这样的: v
几乎从来没有我只想获取某个 Remote 的情况;我总是想要所有的 Remote 。我认为这将是一个足够常见的用例,git 会考虑它(与他们有 pull.rebase true 的方式相同)。 那么,
我正在尝试使用 inarray 但它总是返回 true?有任何想法吗? (所有 li 均已显示) $("#select-by-color-list li").hide(); // get the se
我正在尝试为我公司的开发环境设置过期网址。我们使用 lighttpd在此环境中提供上传的文件,我发现 these docs这似乎相当有希望。 问题是我似乎根本无法让它工作,而且我有点不知所措,试图找出
我无法让“文件夹”外部变量工作。我总是得到[:]。 我正在 Windows 下的 Grails 上进行开发(这就是为什么外部配置文件看起来像 file:C:\path\to/file)。 我在另一个项
这个问题是出于对 PL 如何工作的好奇,而不是其他任何事情。 (它实际上是在查看与 Haskell 不同的 SML 时想到的,因为前者使用按值调用 - 但我的问题是关于 Haskell。) Haske
我有一个高速缓存内存模块,我希望它是可字寻址的,但有字节的写使能信号。 always @ (posedge clk) begin //stuff... if(write) begin
我正在处理一些代码,其中一个对象“foo”正在创建另一个对象对象“bar”,并向其传递一个Callable。之后 foo 将返回bar,然后我希望 foo 变得无法访问(即:可用于垃圾收集)。 我最初
我已将我的程序与此方法相关联: public static void CreateFileAssociation(string extension, string key, string descri
所以我正在进行目录遍历,但我无法让 opendir 按照我想要的方式工作。它总是无法打开我发送的目录,它给出了一些未知的错误。我通常传入 argv[1],但我放弃了,只是开始硬编码路径。 char *
这个问题在这里已经有了答案: How do I compare strings in Java? (23 个回答) 关闭 9 年前。 出于某种原因,我的(基本)程序总是打印我为 else 语句保留的
我不想冒为此提出破解的风险,因为它涉及 datetime 对象。基本上,我想按如下方式进行转换: 2010-04-21 06:37:53 -> 2010-04-21 06:40:00 2010-08-
我正在用 C 语言玩文件 I/O。我正在尝试使用 fgets 从一个文件中读取数据并将其输出到另一个文件。问题是它总是返回 NULL,因此没有任何内容被复制到输出文件中。这是我的代码: #includ
class MyClass { // empty class with no base class }; int main() { MyClass* myClass = new MyC
我是一名优秀的程序员,十分优秀!