我试图使用<sys/socket.h>
和Qt Creator GUI构build器在c ++中创build一个服务器示例,但是在程序的套接字层上会出现两个奇怪的行为。 首先,我运行服务器,但在我使用telnet
连接到它的第一次尝试是立即closures
尝试127.0.0.1 …
连接到127.0.0.1。
转义字符是'^]'。
外部主机closures连接。
当我第二次尝试连接时,它工作,terminal等待我的input。 第二件事是当我closures连接。 如果我刚刚重新运行,在几分钟之内,程序停止bind
退出和返回:
绑定时出错:地址已被使用
所以我想可能是一个连接正在举行后,我打破它使用函数调用onCortarConexao()
或只是停止debugging器。 无论如何,我错过了什么?
我的代码:
#include "mainwindow.h" #include <QApplication> int main(int argc, char *argv[]) { QApplication a(argc, argv); MainWindow w; w.show(); return a.exec(); }
在MainWindow.cpp上:
void MainWindow::on_pushButton_clicked() { socket1 = new MSocket(); socket1->start(); } void MainWindow::on_pushButton_3_clicked() { socket1->onCortarConexao(); }
套接字类:
#ifndef MSOCKET_H #define MSOCKET_H #include <QString> #include <QObject> #include <QThread> #include <QList> #include <string.h> #include <stdio.h> #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <unistd.h> #define SERVER_BUFFER 4096 #define PORTRUN 15000 class MSocket : public QThread { public: MSocket(); void error(char *msg); void onCortarConexao(); private: int sockfd; int newsockfd; int portno; int clilen; int n; char buffer[SERVER_BUFFER]; struct sockaddr_in serv_addr, cli_addr; u_short port; void run(); }; #endif // MSOCKET_H
套接字实现:
void MSocket::run() { sockfd = socket(AF_INET, SOCK_STREAM, 0); if (sockfd < 0){ error("ERROR opening socket"); exit(-1); } bzero((char *) &serv_addr, sizeof(serv_addr)); portno = PORTRUN; serv_addr.sin_family = AF_INET; serv_addr.sin_port = htons(portno); serv_addr.sin_addr.s_addr = INADDR_ANY; fprintf(stdout,"Iniciando servidor.."); if (bind(sockfd, (struct sockaddr *) &serv_addr, sizeof(serv_addr)) < 0){ error("ERROR on binding"); exit(-1); } listen(sockfd,5); newsockfd = accept(sockfd, (struct sockaddr *) &cli_addr, (socklen_t*) sizeof(cli_addr)); if (newsockfd < 0){ error("ERROR on accept"); exit(-1); } bzero(buffer,SERVER_BUFFER); n = read(newsockfd,buffer,SERVER_BUFFER-1); if (n < 0) error("ERROR reading from socket"); printf("Here is the message: %s",buffer); n = write(newsockfd,"I got your message",18); if (n < 0) error("ERROR writing to socket"); } void MSocket::onCortarConexao(){ printf("Encerrando socket"); close(newsockfd); close(sockfd); }
完整的代码位于: https : //github.com/FabioNevesRezende/BasicCppServer
编辑1:
所以, 这是telnet和我的Qt Server应用程序之间通信的数据包列表,它可以在WireShark (.pcapng文件)中看到。 它包含11帧。 第一个是第一个telnet,当它被closures的时候。 就像在应用程序发送[FIN, ACK]
帧4和5上[FIN, ACK]
服务器通过closures连接来响应它。 帧7,8,9是连接的第二次尝试,帧10和11是我发送abc
到服务器。 如在打印屏幕中:
问题是我不知道为什么应用程序正在发送这个FIN
和它在代码中的位置。
与setsockopt
一起使用SO_REUSEADDR
:
optval = 1; setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof optval);
这允许端口被其他套接字重复使用,并绕过您正在address already in use
的address already in use
。
以前的答案表明如何正确处理读取以及如何使用SO_RESUEADDR。 我想指出为什么你的程序只能收到一条消息。 因为你的run函数没有循环,所以只能执行一次,然后退出。 接受客户端的请求后,建立连接,然后进入一个循环等待客户端的输入,并在读取函数返回时写回。
这是我发现的第一个严重问题:
n = read(newsockfd,buffer,SERVER_BUFFER-1); if (n < 0) error("ERROR reading from socket"); printf("Here is the message: %s",buffer);
没有处理read
返回零。 %s
格式说明符仅适用于字符串。 您不能将其用于任何未经检查的数据。
在unix操作系统上,系统调用有时会失败,错误号不会设置为EINTR。 这意味着一个信号被传送到中断系统调用的进程,并且应该重试。 尝试这样的事情:
while ((n = read(newsockfd,buffer,SERVER_BUFFER-1) < 0) { if (errno != EINTR) break; } if (n < 0) error("ERROR reading from socket");
写入的调用也应该完成。