QT client to C server
-
Mai am o problema acum. Dupa ce trec de login, aplicatia mea ofera un meniu cu mai multe posibilitati. In momentul in care dau click pe o optiune, as vrea ca clientul meu sa transmita date serverului prin acelasi socket pe care l-am folosit la login (motivul e ca am foarte multe optiuni si nu stiu cat de eficient e sa fac socket nou de fiecare data). Singura modalitate pe care am reusit sa o implementez consta in
@ pSocket = new QTcpSocket(this);
pSocket->connectToHost("127.0.0.1", 1234);
if (pSocket->waitForConnected())
{
char buffer[]="text";
int len = strlen(buffer);
buffer[len] = '\n';
buffer[len+1] = '\0';
pSocket->write(buffer);
pSocket->flush();
}@
E total ineficient pentru ca imi face conexiuni noi de fiecare data cand transmit date. Am incercat sa imi fac clasa fotbal sa mosteneasca clasa login(unde e socketul), dar nu am avut succes. Vreo idee? -
bq. Dupa ce trec de login
Sa inteleg ca ai rezolvat problema ? Daca da, poti sa zici in forum care a fost solutia pentru ca alti useri sa poata vedea ?
bq. as vrea ca clientul meu sa transmita date serverului prin acelasi socket pe care l-am folosit la login
Da este posibil sa folosesti aceeasi conexiune daca nu o inchizi, fie in server, fie in client, deci inchide socket (close, shutdown) doar cand clientul se opreste.
bq. buffer[len+1] = '\0';
La linia acesta la un moment dat o sa iti crape clientul. De ce nu folosesti Qt sau macar std:: string class ? Alta mica sugestie ii sa scrii prima data marimea sirului pt. ca in server sa stii cate caractere sa citesti.
-
bq. La linia acesta la un moment dat o sa iti crape clientul.
Sorry, my bad, am fost furat de peisaj :-)
Nu o sa crape, dar buffer [5] already are valoarea '\0' nu trebuie sa mai scrii manual. La linia 7 faci ca buffer sa aibe valoare 't', 'e', 'x', '\n'. Asta doresti ? -
Nu e chiar o rezolvare, de vina era serverul meu C asa ca am decis sa fac altul in QT iar acum functioneaza fara sa fac modificari. Partea cu write spre server este pur simbolica pentru moment. Multumesc pentru sesizarea cu 't' 'e' 'x' '\n', nu asta doream sa se intample. Totusi, cum folosesc acelasi socket? Momentan eu am in clasa Login declarat socketul pSocket. Vreau sa il folosesc si in clasele fotbal si tenis, dar primesc eroare de genul pSocket was not declared in this scope. Partea de cod de mai sus este functionala dar eu nu vreau sa se conecteze la host, intrucat asta s-a intamplat deja in login. Vreau doar sa folosesc acelasi socket pentru transmitere de date.
-
Buna, scuze pt. intarziere, am fost f. ocupat zilele acestea.
bq. Vreau sa il folosesc si in clasele fotbal si tenis, dar primesc eroare de genul pSocket was not declared in this scope.
Exista mai multe posibilitati, depinde de softul pe care vrei sa il implementezi. Poti deriva clasele fotbal si tenis din login, si o sa ai acces la socketul din login. Poti sa pasezi login object in constructorii de la tenis, fotbal. Poti avea login class ca si singleton pattern, etc ...
-
Din pacate cunostintele mele de OOP sunt reduse, deci as vrea sa imi recomanzi cea mai simpla solutie. Am incercat sa derivez, dar nu mi-a mers.
Codul arata asa@class Footbal : public QDialog, public Login@
Primesc eroarea:'QObject' is an ambiguos base of 'Footbal' - de 4 ori. Si un warning:
Class Footbal inherits from two QObject subclasses QDialog and Login. This is not supported!public QDialog era pus inca din momentul in care am realizat interfata si am creat legaturile on_pushbutton_clicked.
-
Am scos-o la capat cu chestia asta. Acum stiu ca mesajele ajung de la client la server si de la server la client. Problema e urmatoarea: eu vreau sa transmit date destul de multe de la server la clienti, inclusiv \n-urile. Pentru a le afisa in client folosesc QMessagebox-uri. Nu am reusit sa primesc in client decat cate un singur rand. Intrebarea mea e: nu poate Messagebox-ul sa suporte randuri multiple? Sau cum pot transmite date mari de la server la client.
O portiune de cod in care incerc sa transmit date la client:
@ QByteArray Data = socket->readAll(); // citirea informatiei din socket;
qDebug() << socketDescriptor << "Data in: " << Data; // scrierea informatiei de la client
QString Data2;
Data2 = "\n1 Dinamo\n2 Steaua\n 3Vaslui\n";
socket->write(Data); //test, sa vad daca merge sa scriu inapoi ce am primit
socket->flush();
socket->write(Data2.toUtf8());
socket->flush();@Partea interesanta e ca de fiecare data cand apas butonul respectiv primesc doar pana la \n - apas odata, primesc 1 Dinamo, mai apas odata, primesc 2 Steaua etc.
-
bq. nu poate Messagebox-ul sa suporte randuri multiple?
QMessageBox suporta randuri multiple, dar nu aici cred ca este problema ta
Din cate vad trimiti de 2 ori aceleasi date, odata la linia 4 si odata la linia 7
La client ce primesti daca chemi
@
int receivedBytesSize = socket->readAll()->size();
@Ce valoare are "receivedBytesSize" ?
-
@if(pSocket->waitForConnected())
{
int recvBS = pSocket->readAll().size();
qDebug () << recvBS;
}@
Ciudat.
Cand apas prima oara pe butonul respectiv imi scrie 0. Dar daca apas iar imi da 28. Daca mai apas in continuare, tot 28. Deci 28 cred ca e marimea. Dar nu-s lamurit ce inseamna 0 ala intai. Nu as vrea sa am pierderi de dateCum pot face sa am mai mult spatiu? Vreau sa trimit chiar poate chiar si 2048 caractere
-
bq. Cand apas prima oara pe butonul respectiv imi scrie 0. Dar daca apas iar imi da 28.
Poti sa scrii exact ce trimiti de la server ?
Iar codul de mai sus pune-l intr-un slot connectat la readyRead signal emis de "pSocket"bq. Nu as vrea sa am pierderi de date
Nu o sa ai niciodata pierderi de date. TCP protocol tocmai asta este, garanteaza ca toate packets ajung la destinatie si in ordinea in care sunt trimise.
-
@void Thread::readyRead()
{
QByteArray Data = socket->readAll(); // citirea informatiei din socket;
qDebug() << socketDescriptor << "Data in: " << Data; // scrierea informatiei de la client
QString Data2;
Data2 = "\n1 Dinamo\n2 Steaua\n 3Vaslui\n";
socket->write(Data2.toUtf8());
socket->flush();
}@asta e functia din server.
Acum rezolv si cu readyRead-ul in client si postez imediat
-
bq. Am testat si primesc doar 28 in output.
... deci functioneaza bine acuma (din cate inteleg)
bq. In server o sa vreau sa citesc din fisier si sa trimit la client
Poti sa trimiti date la client citite de oriunde, de pe hdd, din alt socket, din memorie, etc ... poti sa creezi orice pachet de date