我的专属QQ (三) 附源码

 要源码的朋友太多了,满眼的邮箱地址,我很头疼。鉴于现在CSDN首页的Qt应用大赛正在火热进行中,我干脆把源码奉献出来,给大家参考一下好了。不过,这不是我一个人的劳动成果,我一直认为技术领域的最高境界是分享,所以希望得到你的尊重。

       有什么好的意见和建议欢迎你提,但是请注意语气。我写博客的目的有两个,一是记录,二是分享。我记录我的学习历程,分享给大家。我不是神,我想神也不能精通各个技术领域,不是最好的方案你就冷嘲热讽,于情于理都不太合适吧。我这又不是出书,你掏腰包买了看了觉得不好,骂几句才痛快。我自己写我的自己博客,你发现瑕疵,是好事,欢迎你提出来,哪来那么大火气我就不懂了。你能写出更好的,我向你学习。如若不然,在写出作品之前,请别带着一副大爷的嘴脸耍青皮。

   

       请原谅我说了那么多前言范er的废话,我最后再说说关于QThread 的用法,tcpSocket起一个线程去读取用户注册信息这么一档子事。

      

   

 

 我的专属QQ (三) 附源码

 

 

        线程,就是这么个模子:

       

[cpp]  view plain copy
  1. TcpConThread::TcpConThread(int socketDescriptor, QObject *parent)  
  2.         : QThread(parent), socketDescriptor(socketDescriptor)  
  3. {  
  4.     ...  
  5. }  
  6.   
  7. void TcpConThread::run()  
  8. {  
  9.    ...  
  10. }  

 

备个注吧:QThread类提供了与系统无关的线程。 

QThread代表在程序中一个单独的线程控制,在多任务操作系统中,它和同一进程中的其它线程共享数据,但运行起来就像一个单独的程序一样。它不是在main()中开始,QThread是在run()中开始运行的。你继承run()并且在其中包含你的代码。例如:

[cpp]  view plain copy
  1. class MyThread : public QThread {  
  2.   
  3.     public:  
  4.   
  5.         virtual void run();  
  6.   
  7.     };  
  8.   
  9.     void MyThread::run()  
  10.     {  
  11.         forint count = 0; count < 20; count++ ) {  
  12.             sleep( 1 );  
  13.             qDebug( "Ping!" );  
  14.         }  
  15.     }  
  16.   
  17.     int main()  
  18.     {  
  19.         MyThread a;  
  20.         MyThread b;  
  21.         a.start();  
  22.         b.start();  
  23.         a.wait();  
  24.         b.wait();  
  25.     }  

    

void QThread::run () [纯虚 保护] 
这个方法是纯虚的,并且为了能够做有用的工作必须在继承类中被重新实现。这个方法的返回将会结束线程的执行。 

void QThread::start () 
通过调用run()(必须在QThread子类中重新实现来包含你的代码)开始这个线程的执行。如果你试图开始一个已经运行的线程,这个调用将一直等待,直到这个线程完成,然后再重新开始这个线程。

void QThread::exit () [静态]

结束调用线程的执行并且唤醒任何等待它终止的线程。

 

bool QThread::wait ( unsigned long time = ULONG_MAX )

这将提供和POSIX pthread_join相似的功能。一个线程调用了它将会一直阻塞,知道下述条件之一满足时继续:

1.这个QThread对象所关联的线程已经结束执行(比如,当它从run()中返回)。如果线程完成,这个函数将返回真。如果线程还没有开始呢,这个函数也返回真。

2.time毫秒过去了。如果time是ULONG_MAX(默认值),然后等待将永远不会超时(线程必须从run()中返回)。如果等待时间到了,这个函数将返回假。

 

 

 

想在线程里干什么,就在run()里写什么,在这里的实现很简单,内容都在on_Ready_Read()的实现里

 

[cpp]  view plain copy
  1. void TcpConThread::run()  
  2. {  
  3.   
  4.     tcpSocket = new QTcpSocket;  
  5.   
  6.     connect(tcpSocket, SIGNAL(readyRead()), this, SLOT(on_Ready_Read()));  
  7.   
  8.     if (!tcpSocket->setSocketDescriptor(socketDescriptor))  
  9.     {  
  10.         emit error(tcpSocket->error());  
  11.         return;  
  12.     }  
  13.         exec();  
  14.   
  15. }  

这个exec() 就是把整个进程空间换成要执行的那个程序的进程空间,说白了就把自己换成别人。

 

[cpp]  view plain copy
  1. void TcpConThread::on_Ready_Read()  
  2. {  
  3.     /*QString strLogin = tcpSocket->readAll(); 
  4.     QStringList strListUser = strLogin.split("|"); 
  5.     QString id = strListUser.at(0); 
  6.     QString password = strListUser.at(1);*/  
  7.   
  8.     db = new SqliteDB;  
  9.   
  10.     QString ip = tcpSocket->peerAddress().toString();  
  11.     qint16 port = tcpSocket->peerPort();  
  12.   
  13.     QByteArray block = tcpSocket->readAll();  
  14.     QDataStream in(&block, QIODevice::ReadOnly);     //QDataStream in(tcpSocket);  
  15.     quint16 dataGramSize;  
  16.     QString msgType;  
  17.     in >> dataGramSize >> msgType;  
  18.   
  19.     if ( "MSG_CLIENT_USER_REGISTER" == msgType )  
  20.     {  
  21.         QString id;  
  22.         QString password;  
  23.         QString name;  
  24.         in >> id >> password >> name;  
  25.   
  26.         if ( 0 == db->insertNewUser( id, password, name, ip, QString::number(port)) )  
  27.         {  
  28.             QMessageBox::warning(NULL, tr("提示"), tr("该号码已被注册."));  
  29.             QString msgType = "MSG_ID_ALREADY_EXIST";  
  30.             QByteArray block;  
  31.             QDataStream out(&block, QIODevice::WriteOnly);  
  32.             out.setVersion(QDataStream::Qt_4_6);  
  33.             out << (quint16)0 << msgType;  
  34.             out.device()->seek(0);  
  35.             out << (quint16)(block.size() - sizeof(quint16));  
  36.             tcpSocket->write(block);  
  37.         }  
  38.         else  
  39.         {  
  40.             QByteArray block;  
  41.   
  42.             QDataStream out(&block, QIODevice::WriteOnly);  
  43.             out.setVersion(QDataStream::Qt_4_6);  
  44.             QString msgType = "MSG_REGISTER_SUCCESS";  
  45.             out << (quint16)0 << msgType;  
  46.             out.device()->seek(0);  
  47.             out << (quint16)(block.size() - sizeof(quint16));  
  48.             tcpSocket->write(block);  
  49.         }  
  50.     }  
  51.     else if ( "MSG_USER_LOGIN" == msgType )  
  52.     {  
  53.         QString id;  
  54.         QString password;  
  55.         in >> id >> password;  
  56.         db->getUserInfo(id);  
  57.   
  58.         if (db->strListUser.isEmpty())        //MSG_ID_NOTEXIST  
  59.         {  
  60.              QMessageBox::critical(NULL, tr("提示"), tr("没有名字") );  
  61.             QByteArray block;  
  62.             QDataStream out(&block, QIODevice::WriteOnly);  
  63.             out.setVersion(QDataStream::Qt_4_6);  
  64.             QString msgType = "MSG_ID_NOTEXIST";  
  65.             out << (quint16)0 << msgType;  
  66.             out.device()->seek(0);  
  67.             out << (quint16)(block.size() - sizeof(quint16));  
  68.             tcpSocket->write(block);  
  69.   
  70.         }  
  71.         else if(db->strListUser.at(1) != password)        //MSG_PWD_ERROR  
  72.         {  
  73.   
  74.             QByteArray block;  
  75.             QDataStream out(&block, QIODevice::WriteOnly);  
  76.             out.setVersion(QDataStream::Qt_4_6);  
  77.             QString msgType = "MSG_PWD_ERROR";  
  78.             out << (quint16)0 << msgType;  
  79.             out.device()->seek(0);  
  80.             out << (quint16)(block.size() - sizeof(quint16));  
  81.             tcpSocket->write(block);  
  82.         }  
  83.         else if (db->strListUser.at(1) == password )  
  84.         {  
  85.             if ((db->strListUser.at(3)) == "1")          //MSG_LOGIN_ALREADY  
  86.             {  
  87.   
  88.                 QByteArray block;  
  89.                 QDataStream out(&block, QIODevice::WriteOnly);  
  90.                 out.setVersion(QDataStream::Qt_4_6);  
  91.                 QString msgType = "MSG_LOGIN_ALREADY";  
  92.                 out << (quint16)0 << msgType;  
  93.                 out.device()->seek(0);  
  94.                 out << (quint16)(block.size() - sizeof(quint16));  
  95.                 tcpSocket->write(block);  
  96.   
  97.   
  98.             }  
  99.             else            //MSG_LOGIN_SUCCESS  
  100.             {  
  101.                 QByteArray block;  
  102.                 QDataStream out(&block, QIODevice::WriteOnly);  
  103.                 out.setVersion(QDataStream::Qt_4_6);  
  104.                 QString msgType = "MSG_LOGIN_SUCCESS";  
  105.                 out << (quint16)0 << msgType;  
  106.                 out.device()->seek(0);  
  107.                 out << (quint16)(block.size() - sizeof(quint16));  
  108.                 tcpSocket->write(block);  
  109.                 //login success, update database  
  110.                 db->updateUserLogStat(id, "1");  
  111.                 db->updateUserIp(id,tcpSocket->peerAddress().toString());  
  112.             }  
  113.   
  114.         }  
  115.     }  

还有请注意, QCoreApplication::exec() 必须从主线程(执行 main() 方法的线程)调用, 不从 QThread 调用。 在GUI应用程序,主线程也被称为GUI线程,因为它是唯一允许执行GUI相关操作的线程。

 

   

 

困死了还有一大堆工作没有做,整个工程的源码debug版 , release版都打包好上传至CSDN资源,本地没有Qt库也可以运行体验一下,很囧的是,我只能上传20M以下的资源,只好缩减了一些UI的特效显示,唉……只是些UI资源图片的删减无大碍,免费0积分下载:自定义QQ

 

你可能感兴趣的:(我的专属QQ (三) 附源码)