crossplatform.ru

Здравствуйте, гость ( Вход | Регистрация )


  Ответ в QDatStream
Введите ваше имя
Подтвердите код

Введите в поле код из 6 символов, отображенных в виде изображения. Если вы не можете прочитать код с изображения, нажмите на изображение для генерации нового кода.
 

Опции сообщения
 Включить смайлы?
Иконки сообщения
(Опционально)
                                
                                
  [ Без иконки ]
 


Последние 10 сообщений [ в обратном порядке ]
gpepsi Дата 19.9.2011, 10:01
 
Цитата(BRE @ 19.9.2011, 10:52) *
А ты состояние проверяй:
SocketState QAbstractSocket::state () const


ща попробую

Цитата(BRE @ 19.9.2011, 10:52) *
Можно в начале отправлять размер пакета и ждать пока придут все данные читая bytesAvailable, а только после этого вычитывать строку из потока.

а буфера хватит ?
BRE Дата 19.9.2011, 9:52
 
Цитата(gpepsi @ 19.9.2011, 9:34) *
у клиента он проходит. Но сервер не подключен к сигналу, а просто проверяет isValid - вот он проболжает возвращать true после закрытия клиента.

А ты состояние проверяй:
SocketState QAbstractSocket::state () const

Цитата(gpepsi @ 19.9.2011, 9:34) *
З.Ы. Может тогда унаследоваться от QDataStream и переопределить readRawData & writeRawData, чтоб остались все его возможности

А что это даст?
Тебе нужен просто цикл добавить и ждать когда придут все данные для чтения хотя-бы одной строки.
Можно в начале отправлять размер пакета и ждать пока придут все данные читая bytesAvailable, а только после этого вычитывать строку из потока.
gpepsi Дата 19.9.2011, 8:34
 
Цитата(BRE @ 19.9.2011, 9:28) *
А с этим я не понял, что не приходит? Сигнал disconnected?


у клиента он проходит. Но сервер не подключен к сигналу, а просто проверяет isValid - вот он проболжает возвращать true после закрытия клиента.

З.Ы. Может тогда унаследоваться от QDataStream и переопределить readRawData & writeRawData, чтоб остались все его возможности
BRE Дата 19.9.2011, 8:28
 
Цитата(gpepsi @ 19.9.2011, 8:51) *
ну тогда нужно держать свой буфер накопления, а хотелось возложить это на QDataStream. Ему никак нельзя сказать, чтоб он читал покак не придет все ?

Не буду утверждать, а проверить времени сейчас нет, но скорее всего будет использоваться буфер сокета. Тебе нужно будет дождаться прихода всех пакетов для чтения строки, а пока это не так QDataStream будет возвращать пустую строку.

Цитата(gpepsi @ 19.9.2011, 8:51) *
ну тоже самое - самому контролировать. Просто я думал, что для TCP это сделано (какой-нибудь QTcpStream)

А тут да - нужно будет заводить свой буфер.

Цитата(gpepsi @ 19.9.2011, 8:51) *
P.S. А что по поводу закрытия сокетов ?

А с этим я не понял, что не приходит? Сигнал disconnected?
gpepsi Дата 19.9.2011, 7:51
 
Цитата(BRE @ 19.9.2011, 8:37) *
Можно пробовать читать из потока и если вернулась пустая строка, то ждать прихода следующего пакета и пробовать читать снова.

ну тогда нужно держать свой буфер накопления, а хотелось возложить это на QDataStream. Ему никак нельзя сказать, чтоб он читал покак не придет все ?

Цитата(BRE @ 19.9.2011, 8:37) *
Можно писать raw-строки без использования QDataStream и в качестве конца строки использовать "\r\n" (как это делается в http, ftp, ...). Тогда каждую порцию строку можно дописывать в буфер и контролировать маркер конца строки, если он получен, то вырезаем ее из буфера и обрабатываем.

ну тоже самое - самому контролировать. Просто я думал, что для TCP это сделано (какой-нибудь QTcpStream)


P.S. А что по поводу закрытия сокетов ?
BRE Дата 19.9.2011, 7:37
  При серилизации QString вначале записывается ее длина, поэтому QDataStream может определить пришла она полностью или нет. Если она пришла не полностью, то ему не остается ничего другого, кроме возвращения пустой строки при чтении.
Можно пробовать читать из потока и если вернулась пустая строка, то ждать прихода следующего пакета и пробовать читать снова.
Можно писать raw-строки без использования QDataStream и в качестве конца строки использовать "\r\n" (как это делается в http, ftp, ...). Тогда каждую порцию строку можно дописывать в буфер и контролировать маркер конца строки, если он получен, то вырезаем ее из буфера и обрабатываем.
gpepsi Дата 19.9.2011, 5:10
 
Цитата(BRE @ 17.9.2011, 22:28) *
Ну так QDataStream так и читает.

откуда тогда пустые данные

Цитата(BRE @ 17.9.2011, 22:28) *
Расскажи подробней что передается/принимается и как это выглядит в коде.

просто строка длинной более MTU

Сервер в методе incomingConnection
    while(socket.isValid())
    {
        if (socket.waitForReadReady())
        {
            ...
        }
    }
    socket.close();


Клиент
    while(socket.isValid())
    {
        QDataStream stream(&socket);
        stream << QString("...");
        if (socket.waitForReadReady())
        {
            QString data;
            stream >> data;
            ...
        }
    }



З.Ы. Есть еще проблема. Когда сервер рвет соединение, то клиенту сразу приходит сигнал.
А вот если клиент рвет, со серверу пофигу. Вот тестовый код
    #include <QtCore/QCoreApplication>
    #include <QTcpServer>
    #include <QTcpSocket>
    #include <QCoreApplication>
    
    class Client {
    private:
        QTcpSocket m_socket;
    
    public :
        Client () {}
        ~Client() {}
    
        void connect (void)
        {
            m_socket.connectToHost(QHostAddress::LocalHost, 7777);
            if (!m_socket.waitForConnected())
            {
                QT_THROW(m_socket.errorString());
            }
            qDebug("CONNECTED !!!");
        }
    
        void disconnect(void)
        {
            m_socket.close();
            qDebug("DISCONNECTED !!!");
        }
    
        bool isValid(void) const { return m_socket.isValid(); }
    };
    
    class Server : public QTcpServer {
    private:
        Client m_client;
    
        virtual void timerEvent ( QTimerEvent * event )
        {
            static uint counter = 0;
            switch (++counter)
            {
                case 1 : m_client.connect();    break;
                case 2 : m_client.disconnect(); break;
            }
        }
        virtual void incomingConnection( int session )
        {
            QTcpSocket socket(this);
            socket.setSocketDescriptor(session);
    
            while (socket.isValid())
            {
                Q_ASSERT(m_client.isValid());   // Здесь сработает ASSERT, но socket.isValid() :)
                qApp->processEvents();
            }
            socket.close();
        }
    
    public :
        Server ()
        {
            if (!listen(QHostAddress::Any, 7777))
            {
                QT_THROW(QTcpServer::errorString());
            }
            startTimer(100);
        }
        virtual ~Server() {}
    };
    
    int main(int argc, char *argv[])
    {
        QCoreApplication a(argc, argv);
        Server server;
        return a.exec();
    }
BRE Дата 17.9.2011, 21:28
 
Цитата(gpepsi @ 17.9.2011, 22:22) *
Есть ли что стандартное, для чтения потока? То есть читать пока читается либо пока не порвется соединение.

Ну так QDataStream так и читает.
Расскажи подробней что передается/принимается и как это выглядит в коде.
gpepsi Дата 17.9.2011, 21:22
  Есть сокет. Попытался читать из него QDataStream.
На больших данных operator >> возвращает пустые данные.
Есть предположение, что QDataStream читает пока читается, но если данные приходят кусками, то он не успевает все вычитывать.

Есть ли что стандартное, для чтения потока? То есть читать пока читается либо пока не порвется соединение.
Просмотр темы полностью (откроется в новом окне)
RSS Текстовая версия Сейчас: 29.3.2024, 2:42