qtcpserver用法_qt tcpserver

qtcpserver用法_qt tcpserver【Qt服务器与多线程使用】        ~~~~~~~~        QTcpServer致命缺点就是单线程!!要想实现多线程则需要继承该类之后重写incomingConnection函数,在该函数中将socketDescripto…

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

QTcpServer和QTcpSocket使用详解


1、基本使用方法

QTcpServer和QTcpSocket的使用是密不可分的,所以两者一块演示使用方法。

QTcpServer常用信号:

  • newConnection()信号,该信号用于处理新接入的连接

QTcpSocket常用信号:

  • disconnected():断开连接后会触发该信号,一般在该信号的槽函数内处理内存清理释放等工作
  • readyRead():数据到达信号,在该信号的槽函数内处理数据接收

补充:如果想要获取远程接入的IP地址端口等信息,可以使用QTcpSocket的peerAddress()函数获取IP地址,peerPort()获取端口。

基于QTcpServer的TCP服务器多接入echo代码demo:

#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include <QTcpServer>
#include <QTcpSocket>
#include <QDebug>

namespace Ui { 
   
class Widget;
}

class Widget : public QWidget
{ 
   
    Q_OBJECT

public:
    explicit Widget(QWidget *parent = 0);
    ~Widget();

private:
    Ui::Widget *ui;

private:
    QTcpServer* tcpServer;

private slots:
    void slotReadyRead();
    void slotNewConnected();
};

#endif // WIDGET_H

#include "widget.h"
#include "ui_widget.h"

Widget::Widget(QWidget *parent) :
    QWidget(parent),
    ui(new Ui::Widget)
{ 
   
    ui->setupUi(this);

    tcpServer = new QTcpServer(this);
    QObject::connect(tcpServer, SIGNAL(newConnection()), this, SLOT(slotNewConnected()));
    tcpServer->listen(QHostAddress::Any, 10126);
}

Widget::~Widget()
{ 
   
    delete ui;
}


void Widget::slotNewConnected()
{ 
   
    while(tcpServer->hasPendingConnections())
    { 
   
        QTcpSocket* socket = tcpServer->nextPendingConnection();
        QObject::connect(socket, SIGNAL(readyRead()), this, SLOT(slotReadyRead()));
        QObject::connect(socket, SIGNAL(disconnected()), socket, SLOT(deleteLater()));
        qDebug("[%s:%d] connected", socket->peerAddress().toString().toStdString().c_str(), socket->peerPort());
    }
}

void Widget::slotReadyRead()
{ 
   
    QTcpSocket* socket = qobject_cast<QTcpSocket*>(sender());  // 取得信号发送者对象
    socket->write(socket->readAll());                          // 将数据在发送回去
}

2、基于多线程的使用

         ~~~~~~~~         处理网络的数据一般不会是单线程处理,因为要更新UI界面且还有任务,不可能让tcpSocket死等接收,正常情况下通过QTcpServer获得新接入的客户端QTcpSocket对象来进行收发数据操作,新接入一个连接就创建一个新的线程去处理,但是这里有一个问题就是 QTcpSocket是不支持跨线程调用的,哪怕是以引用或者指针的形式传入到子线程中,也是报错跨线程调用!

         ~~~~~~~~         要想实现多线程则需要继承QTcpServer去重写incomingConnection函数,在该函数中将socketDescriptor参数用信号把他发射出来,或者直接在该函数中创建自己的任务处理线程,在线程中使用QTcpSocket类的setsetSocketDescriptor函数初始化一个socket就可以了。示例代码如下:

#ifndef QMULTITCPSERVER_H
#define QMULTITCPSERVER_H

#include <QObject>
#include <QTcpServer>
#include <QTcpSocket>
#include <QHostAddress>

class QMultiTcpServer : public QTcpServer
{ 
   
    Q_OBJECT

public:
    QMultiTcpServer(QObject* parent = nullptr);
    virtual ~QMultiTcpServer();

public:
    bool listen(const QHostAddress &address, quint16 port);

protected:
    void incomingConnection(int socketDescriptor);

signals:
    void newConnection(int socketDescriptor);
};

#endif // QMULTITCPSERVER_H

#include "qmultitcpserver.h"

QMultiTcpServer::QMultiTcpServer(QObject* parent):QTcpServer(parent)
{ 
   

}

QMultiTcpServer::~QMultiTcpServer()
{ 
   

}

void QMultiTcpServer::incomingConnection(int socketDescriptor)
{ 
   
    emit this->newConnection(socketDescriptor);
}

bool QMultiTcpServer::listen(const QHostAddress &address, quint16 port)
{ 
   
    return QTcpServer::listen(address, port);
}

示例代码中,我又新增了一个带参数的newConnected信号,目的是为了把socketDescriptor通过信号发射到槽函数中。

然后在主窗体实例化对象:

    tcpServer = new QMultiTcpServer(this);
    QObject::connect(tcpServer, SIGNAL(newConnection(int)), this, SLOT(slotNewConnected(int)));

对应的槽函数如下:

void Widget::slotNewConnected(int socketDescriptor)
{ 
   
    QTcpSocket* socket = new QTcpSocket(this);
    if(!socket->setSocketDescriptor(socketDescriptor))
    { 
   
        delete socket;
        return;
    }

	//
	// 你可以在这里新建一个处理线程,将socketDescriptor传入处理线程,我这里直接是演示怎么使用
	//

    QObject::connect(socket, SIGNAL(readyRead()), this, SLOT(slotReadyRead()));
    QObject::connect(socket, SIGNAL(disconnected()), this, SLOT(slotDisconnected()));
    qDebug("[%s:%d]: connected.", socket->peerAddress().toString().toStdString().c_str(), socket->peerPort());
}

void Widget::slotReadyRead()
{ 
   
    QTcpSocket* socket = qobject_cast<QTcpSocket*>(sender());  // 取得当前socket对象
    QByteArray data = socket->readAll();
    qDebug("[%s:%d]:%s", socket->peerAddress().toString().toStdString().c_str(), socket->peerPort(), data.toStdString().c_str());
}

void Widget::slotDisconnected()
{ 
   
    QTcpSocket* socket = qobject_cast<QTcpSocket*>(sender());  // 取得当前socket对象
    socket->close();
    qDebug("[%s:%d]: disconnected.", socket->peerAddress().toString().toStdString().c_str(), socket->peerPort());
    delete socket;
}

         ~~~~~~~~         附上多线程tcp服务器的一个demo,代码功能比较简单,就是只有回显功能。客户端接入后新开一个线程处理数据的发送和接收,执行效果如下:
https://github.com/wowyyy/QtThreadServer
在这里插入图片描述


版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/197302.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • @transactional的使用_@transactional注解默认的回滚方式

    @transactional的使用_@transactional注解默认的回滚方式@Transactional是声明式事务管理编程中使用的注解1.添加位置1)接口实现类或接口实现方法上,而不是接口类中。2)访问权限:public的方法才起作用。@Transactional注解应该只被应用到public方法上,这是由SpringAOP的本质决定的。系统设计:将标签放置在需要进行事务管理的方法上,而不是放在所有接口实现类上:只读的接口就不需要事务管…

    2022年9月30日
    3
  • deepfacelab最新版使用教程(deepfacelab训练技巧)

    欢迎进入本教程,本教程不定期更新本文教程内容更新时间为:2019/2,本文最后更新时间为:2019/3/8欢迎进群讨论,我不是群主,群主的各QQ群号码请看:https://deepfakes.com.cn/index.php/资助升级群这几天朱茵换脸杨幂的事件上了热门,我们群不存在该违法问题。然而因为网上的这个事件,2群排队已经可以说排到了明年。群主表示等风波结束了再考虑新群,毕竟群主不想…

    2022年4月17日
    518
  • c语言数组介绍

    c语言数组介绍文章目录一、一维数组的创建和初始化1.数组的创建2.数组的初始化3.一维数组的使用4.一维数组在内存中的存储二、二维数组的创建和初始化1.二维数组的创建2.二维数组的创建3.二维数组的创建4.二维数组在内存中的存储二、数组越界1.数组名是什么?一、一维数组的创建和初始化1.数组的创建数组是一组相同类型元素的集合。数组的创建方式:type_tarr_name[const_n];//type_t是指数组的元素类型//const_n是一个常量表达式,用来指定数组的大小数组创建的实

    2022年7月11日
    23
  • 补码加、减运算规则「建议收藏」

    补码加、减运算规则「建议收藏」在计算机中,通常总是用补码完成算术的加减法运算。其规则是:  [X+Y]补=[X]补+[Y]补,[X-Y]补=[X]补-[Y]补=[X]补+[-Y]补这表明,有了补码表示的被加(减)数和加(减)数,要完成计算补码表示的二数之和或二数之差,只需用二数的补码直接执行加减运算即可,符号位与数值位同等对待,一起参加运算,若运算结果不溢出,即不超出计算机所能表示的范围,则结果的符号…

    2022年4月19日
    52
  • abaqus6.14.4安装_abaqus激活成功教程教程

    abaqus6.14.4安装_abaqus激活成功教程教程密码zo32

    2025年10月18日
    5
  • jq的 on 事件委托 导致多次执行问题

    jq的 on 事件委托 导致多次执行问题

    2021年7月5日
    101

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号