2016-08-14 87 views
5

我正在嘗試使用<sys/socket.h>和Qt Creator GUI構建器在C++中創建一個服務器示例,但是在程序的套接字層發生了兩個奇怪的行爲。首先,我運行服務器,但在第一次嘗試我做連接它用telnet到被imediately關閉關閉第一次連接嘗試的C++套接字

試圖127.0.0.1 ...
連接到127.0.0.1。
轉義字符是'^]'。
由外國主機關閉的連接。

當我嘗試第二次連接,它的工作原理和終端等待我的輸入。第二件事是當我關閉連接時。如果我之後重新運行,在短短的幾分鐘內,程序上bind暫停退出並返回:

錯誤的綁定:地址已在使用中

,所以我想,也許是正在舉行的連接在我使用函數調用onCortarConexao()或者只是停止調試器來分解它之後。無論如何,我錯過了什麼?

我的代碼:

#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(); 
} 

Socket類:

#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:

所以,this是可以graphicaly看到Wireshark的 telnet和我的Qt服務器應用程序之間的通信包的列表( .pcapng文件)。它包含11個幀。 6個第一個來自第一個遠程登錄,當它被立即關閉時。正如在框架4和5上,應用程序發送[FIN, ACK],服務器通過關閉連接來響應它。幀7,8,9是連接的第二次嘗試,幀10和11是我發送abc到服務器的時間。由於在打印屏幕:

sequence of commands

問題是我不知道爲什麼應用程序在它的代碼發送此FIN和地點。

+0

您爲github項目提供的鏈接已損壞。沒有它,我不能告訴你是什麼意思第二次嘗試只是在客戶端再次嘗試或重新啓動服務器。你能澄清嗎? – lasaro

+0

對不起,它再次打開。 – Fabiotk

回答

4

使用SO_REUSEADDRsetsockopt

optval = 1; 
setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof optval); 

這使得端口被其他插座被重用並圍繞你所面臨的問題address already in use得到。

+0

謝謝,它解決了第二個問題。 – Fabiotk

+1

您可能想解釋爲什麼這會使未來的訪問者成爲一個很好的Q + A。 –

3

這裏的第一個嚴重的問題,我發現:

n = read(newsockfd,buffer,SERVER_BUFFER-1); 
if (n < 0) 
    error("ERROR reading from socket"); 

printf("Here is the message: %s",buffer); 

有沒有處理的read返回零。 %s格式說明符僅適用於字符串。您不能將其用於任意未經檢查的數據。

3

在unix操作系統上,系統調用有時可能會失敗,並將錯誤號設置爲EINTR。這意味着一個信號被傳送到中斷系統調用的進程,並且應該重試。試試這樣:

while ((n = read(newsockfd,buffer,SERVER_BUFFER-1) < 0) { 
if (errno != EINTR) 
break; 
} 
if (n < 0) 
    error("ERROR reading from socket"); 

對於寫入的調用應該做同樣的事情。

5

以前的答案表明如何正確處理讀取以及如何使用SO_RESUEADDR。 我想指出你的程序爲什麼只能收到一條消息。因爲你的run函數沒有循環,所以它只能執行一次,然後線程退出。在接受客戶端請求後,建立連接,然後進入一個循環等待客戶端的輸入,並在讀取函數返回後寫回。

+0

好點。我完全錯過了。 –