Qt笔记-QTcpSocket跨线程调用(官方推荐方法,非百度烂大街方法)
TCP服務端的經典案例中有個例子,就是當收到TCP客戶端連接后,線程池直接開一個線程然后把這個socket指針傳到線程里面,依靠新開的線程進程業務處理。
但在Qt里面使用這個方式后,會報一個QTcpSocket不能跨線程調用的問題。
問題描述是這樣的:
QSocketNotifier: Socket notifiers cannot be enabled or disabled from another thread
查了下百度,都是n年前的技術(本博文寫于2021-08-11 09:15:25,此博文發布到網上應該是1個月以后,筆記太多了,每星期公布6篇筆記),用信號與槽機制將業務處理放到新線程中,TCP操作用信號讓以前的線程發送數據,通過這種方式解決問題!但如果這樣做,代碼就不夠優雅了,甚至還有點爛。一個新技術框架和老技術做對比,應該是取其精華去其糟粕,而不是越來越復雜。不然這個新技術將毫無意義。
花了半天時間查閱了官方資料,最后得出了QTcpSocket跨線程調用的方式。
首先繼承QTcpServer重寫incomingConnection(qintptr handle)
?
當這個handle此時套接字的描述符號。把這個傳給線程,子線程生成一個QTcpScoket然后利用setSocketDescriptor使得這個socket獲得到收發數據的能力。
實例如下:
QTcpServer相關:
void HttpsServer::incomingConnection(qintptr socketDescriptor) {QThreadPool::globalInstance()->start(new HttpsThread(socketDescriptor)); }QRunnable相關:
HttpsThread::HttpsThread(qintptr descriptor) {m_socketDescriptor = descriptor; }這個m_socketDescriptor直接為int型就可以了。
這樣run()中調用setSocketDescriptor,進行讀寫操作就不會有跨線程的問題了。
void HttpsThread::run() {QTcpSocket tcpSocket;if(!tcpSocket.setSocketDescriptor(m_socketDescriptor)) {qDebug() << "tcpSocket.setSocketDescriptor(m_socketDescriptor) failed";return;}tcpSocket.write("Hello Wold");tcpSocket.disconnectFromHost();tcpSocket.waitForDisconnected(); }總結
以上是生活随笔為你收集整理的Qt笔记-QTcpSocket跨线程调用(官方推荐方法,非百度烂大街方法)的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Qt工作笔记-WebEngineView
- 下一篇: Qt文档阅读笔记-moc工具使用及其理论