2010-01-11 5 views
22

Ich habe einen sehr einfachen Server/Client-Leistungstest mit boost :: asio unter Windows und es scheint wirklich schlecht zu sein. Ich hoffe, dass ich die Bibliothek nur falsch benutze und würde mich über jeden Rat freuen.Schlechte Boost.ASIO Leistung

Ich habe eine Sitzungsklasse, die eine Nachrichtenlänge schreibt und dann eine Nachricht schreibt, und dann wartet, um eine Nachrichtenlänge zu lesen und dann eine Nachricht zu lesen, und macht dies immer wieder nonstop. Wenn ich es lokal auf meinem eigenen Computer laufe, bekomme ich jedoch eine blitzschnelle Leistung; Wenn ich einen Server auf einem Computer und einen Client auf einem anderen Computer, sogar im selben Netzwerk, ausführe, verlangsamt sich die Leistung und es dauert bis zu 1 Sekunde, bis eine Lese-/Schreiboperation ausgeführt wird.

Der Server Quellcodedatei ist wie folgt:

#include <cstdlib> 
#include <iostream> 
#include <boost/asio.hpp> 
#include <boost/bind.hpp> 

using namespace boost; 
using namespace boost::asio; 
using namespace boost::asio::ip; 
using namespace std; 

class Session { 
    public: 

    Session(io_service& ioService) 
     : m_socket(ioService) {} 

    tcp::socket& GetSocket() { 
     return m_socket; 
    } 

    void StartRead() { 
     m_messageSizeIterator = reinterpret_cast<char*>(&m_messageSize); 
     async_read(m_socket, buffer(m_messageSizeIterator, sizeof(m_messageSize)), 
     bind(&Session::HandleSizeRead, this, placeholders::error, 
     placeholders::bytes_transferred)); 
    } 

    void StartWrite(const char* message, int messageSize) { 
     m_messageSize = messageSize; 
     m_message = new char[m_messageSize]; 
     memcpy(m_message, message, m_messageSize); 
     async_write(m_socket, buffer(&m_messageSize, sizeof(int)), 
     bind(&Session::HandleSizeWritten, this, placeholders::error)); 
    } 

    void HandleSizeRead(const system::error_code& error, 
     size_t bytes_transferred) { 
     if(!error) { 
     m_message = new char[m_messageSize]; 
     async_read(m_socket, buffer(m_message, m_messageSize), 
      bind(&Session::HandleMessageRead, this, placeholders::error, 
      placeholders::bytes_transferred)); 
     } else { 
     delete this; 
     } 
    } 

    void HandleMessageRead(const system::error_code& error, 
     size_t bytes_transferred) { 
     if(!error) { 
     cout << string(m_message, m_messageSize) << endl; 
     async_write(m_socket, buffer(&m_messageSize, sizeof(int)), 
      bind(&Session::HandleSizeWritten, this, placeholders::error)); 
     } else { 
     delete this; 
     } 
    } 

    void HandleSizeWritten(const system::error_code& error) { 
     if(!error) { 
     async_write(m_socket, buffer(m_message, m_messageSize), 
      bind(&Session::HandleMessageWritten, this, placeholders::error)); 
     } else { 
     delete this; 
     } 
    } 

    void HandleMessageWritten(const system::error_code& error) { 
     if(!error) { 
     delete m_message; 
     m_messageSizeIterator = reinterpret_cast<char*>(&m_messageSize); 
     async_read(m_socket, buffer(m_messageSizeIterator, 
      sizeof(m_messageSize)), bind(&Session::HandleSizeRead, this, 
      placeholders::error, placeholders::bytes_transferred)); 
     } else { 
     delete this; 
     } 
    } 

    private: 
    tcp::socket m_socket; 
    int m_messageSize; 
    char* m_messageSizeIterator; 
    char* m_message; 
}; 

class Server { 
    public: 

    Server(io_service& ioService, short port) 
     : m_ioService(ioService), 
      m_acceptor(ioService, tcp::endpoint(tcp::v4(), port)) { 
     Session* new_session = new Session(m_ioService); 
     m_acceptor.async_accept(new_session->GetSocket(), bind(&Server::HandleAccept, 
     this, new_session,asio::placeholders::error)); 
    } 

    void HandleAccept(Session* new_session, const system::error_code& error) { 
     if(!error) { 
     new_session->StartRead(); 
     new_session = new Session(m_ioService); 
     m_acceptor.async_accept(new_session->GetSocket(), bind(
      &Server::HandleAccept, this, new_session, placeholders::error)); 
     } else { 
     delete new_session; 
     } 
    } 

    private: 
    io_service& m_ioService; 
    tcp::acceptor m_acceptor; 
}; 

int main(int argc, char* argv[]) { 
    try { 
    if(argc != 2) { 
     cerr << "Usage: server <port>\n"; 
     return 1; 
    } 
    io_service io_service; 
    Server s(io_service, atoi(argv[1])); 
    io_service.run(); 
    } catch(std::exception& e) { 
    cerr << "Exception: " << e.what() << "\n"; 
    } 
    return 0; 
} 

Und der Client-Code ist wie folgt:

#include <cstdlib> 
#include <cstring> 
#include <iostream> 
#include <boost/bind.hpp> 
#include <boost/asio.hpp> 

using namespace boost; 
using namespace boost::asio; 
using namespace boost::asio::ip; 
using namespace std; 

class Session { 
    public: 

    Session(io_service& ioService) 
     : m_socket(ioService) {} 

    tcp::socket& GetSocket() { 
     return m_socket; 
    } 

    void StartRead() { 
     m_messageSizeIterator = reinterpret_cast<char*>(&m_messageSize); 
     async_read(m_socket, buffer(m_messageSizeIterator, sizeof(m_messageSize)), 
     bind(&Session::HandleSizeRead, this, placeholders::error, 
     placeholders::bytes_transferred)); 
    } 

    void StartWrite(const char* message, int messageSize) { 
     m_messageSize = messageSize; 
     m_message = new char[m_messageSize]; 
     memcpy(m_message, message, m_messageSize); 
     async_write(m_socket, buffer(&m_messageSize, sizeof(int)), 
     bind(&Session::HandleSizeWritten, this, placeholders::error)); 
    } 

    void HandleSizeRead(const system::error_code& error, 
     size_t bytes_transferred) { 
     if(!error) { 
     m_message = new char[m_messageSize]; 
     async_read(m_socket, buffer(m_message, m_messageSize), 
      bind(&Session::HandleMessageRead, this, placeholders::error, 
      placeholders::bytes_transferred)); 
     } else { 
     delete this; 
     } 
    } 

    void HandleMessageRead(const system::error_code& error, 
     size_t bytes_transferred) { 
     if(!error) { 
     cout << string(m_message, m_messageSize) << endl; 
     async_write(m_socket, buffer(&m_messageSize, sizeof(int)), 
      bind(&Session::HandleSizeWritten, this, placeholders::error)); 
     } else { 
     delete this; 
     } 
    } 

    void HandleSizeWritten(const system::error_code& error) { 
     if(!error) { 
     async_write(m_socket, buffer(m_message, m_messageSize), 
      bind(&Session::HandleMessageWritten, this, placeholders::error)); 
     } else { 
     delete this; 
     } 
    } 

    void HandleMessageWritten(const system::error_code& error) { 
     if(!error) { 
     delete m_message; 
     m_messageSizeIterator = reinterpret_cast<char*>(&m_messageSize); 
     async_read(m_socket, buffer(m_messageSizeIterator, 
      sizeof(m_messageSize)), bind(&Session::HandleSizeRead, this, 
      placeholders::error, placeholders::bytes_transferred)); 
     } else { 
     delete this; 
     } 
    } 

    private: 
    tcp::socket m_socket; 
    int m_messageSize; 
    char* m_messageSizeIterator; 
    char* m_message; 
}; 

int main(int argc, char* argv[]) { 
    try { 
    if(argc != 3) { 
     cerr << "Usage: client <host> <port>\n"; 
     return 1; 
    } 
    io_service io_service; 
    tcp::resolver resolver(io_service); 
    tcp::resolver::query query(tcp::v4(), argv[1], argv[2]); 
    tcp::resolver::iterator iterator = resolver.resolve(query); 
    Session session(io_service); 
    tcp::socket& s = session.GetSocket(); 
    s.connect(*iterator); 
    cout << "Enter message: "; 
    const int MAX_LENGTH = 1024; 
    char request[MAX_LENGTH]; 
    cin.getline(request, MAX_LENGTH); 
    int requestLength = strlen(request); 
    session.StartWrite(request, requestLength); 
    io_service.run(); 
    } catch (std::exception& e) { 
    cerr << "Exception: " << e.what() << "\n"; 
    } 
    return 0; 
} 

Jede Hilfe würde geschätzt, danke.


Für meine Zwecke wirklich wirklich kleine Nachrichten zu senden und virtuelle Echtzeit Antworten zu wollen, wandte mich Nagle-Algorithmus deaktivieren aus der Ursache für die schlechte Leistung zu sein.

+0

Haben Sie die Möglichkeiten von etwas ausgeschlossen, was Ihr Router tun könnte, verursacht das Problem? Wie ist die CPU-Auslastung auf jeder Maschine? – bobber205

+0

Die CPU-Auslastung beträgt auf beiden Computern 0. Da der Router das Problem ist, habe ich ein ähnliches Programm geschrieben, ohne ASIO zu verwenden, und es funktionierte sehr schnell. – Kranar

+3

Beginnen Sie mit dem Testen Ihrer Verbindung mit iperf. Dann instrumentieren Sie den gesamten Prozess - Wurde der Sockel erstellt? Ist die Bindung erfolgreich? Hat die Entschlossenheit funktioniert? Hat die Verbindung zum Server funktioniert? Hat die erste Arbeit gesendet? Arbeitet der Erste? Werden bei einem Netzwerk-API-Aufruf Fehler zurückgegeben? Hat etwas länger gedauert als erwartet? Sehen Sie sich den Netzwerkverkehr an. Gibt es eine Firewall auf dem Server? Ist der Nagle-Algorithmus aktiviert? Hat der Server lange gebraucht, um zu antworten? Gibt es eine Verzögerung, die Sie in dem Nicht-Netzwerk-Code im Client nicht erwartet haben? – Permaquid

Antwort

36

Sie müssen die Nagle algorithm ausschalten. Anruf:

m_socket.set_option(tcp::no_delay(true)); 

Wo für Ihren Code geeignet.

+0

+1 für Vorschlag und für die Verknüpfung in Nagle auf Wiki. Allerdings - hätte ich gerne eine Warnung gesehen, dass das Drehen von Nagle wahrscheinlich den Gesamtdurchsatz senken würde. – quixver

+2

@quixver Kein Nagle führt zu mehr Paketen im Netzwerk, wenn Sie Bytes mit einer kleinen Lücke zwischen ihnen senden. Ein Zeitgeber (dh Nagle) zum Zusammenwachsen führt zu weniger Paketen und verbessert so den gesamten Netzwerkdurchsatz. Dies gilt für den interaktiven Tastaturverkehr (z. B. Telnet, SSH) und war vor zwanzig Jahren ein großer Teil des Ethernet-Datenverkehrs. Bei der Programm-zu-Programm-Kommunikation führt Nagle zu einem insgesamt niedrigeren Durchsatz (wie es in der ursprünglichen Frage der Fall war) statt zu einem höheren Durchsatz. Stellen Sie beispielsweise fest, dass die gesamte Nachricht an async_write() übergeben wurde und Sie daher nicht auf das Senden warten müssen. – janm

6

Für meine Zwecke, wirklich sehr kleine Nachrichten zu senden und virtuelle Echtzeit-Antworten zu wünschen, erwies sich die Deaktivierung Nagles Algorithmus als Ursache für die schlechte Leistung.