Qt系统相关

本文目录

  • 1.Qt事件
    • 事件的处理
      • 标签事件
      • 鼠标事件
      • 滚轮事件
      • 按键事件
      • 定时器事件
      • 窗口事件
      • 事件派发器
  • 2.Qt文件操作
    • QFile的基本使用
  • 3.Qt多线程
    • 使用线程
    • 线程锁
      • connect的第五个参数
    • 条件变量和信号量
  • 4.Qt网络编程
    • UDP Socket
    • TCP Socket
      • QTcpServer
      • QTcpSocket
    • HTTP的编写
  • 5.QT多媒体播放
    • 音频播放
    • 视频播放


前言:
大家好啊,😍😍虽然说QT是一套跨平台的c++开发开发框架,但是Qt很多技术都是封装系统的API是是实现的,所以我们就需要去了解哪些是封装的系统的API。😳

1.Qt事件

什么是事件 对于事件着机制,操作系统本身就有这个机制,我们可以理解Qt对操作系统的事件机制进行了封装。那什么是事件呢? 之前我们在学习信号与槽是为了应对用户在进行某些操作之后,对该操作进行处理,通过产生信号调用对应的槽函数。那么所谓的事件也是非常的类似,用户进行的各种操作也会产生事件,我们也可以去对事件进行一些事件处理。可以发现,这两种机制实现的效果基本一样。 由于事件处理比较麻烦,因此Qt进行了进一步的封装,形成了信号槽,通过事件与信号槽,我们都可以完成某种信号或者事件下,对用户的操作进行处理。所以可以认为信号槽的“爸爸”就是事件。 绝大多数情况下我们都是使用信号槽进行交互的,但是面对一些特殊情况,我们还是需要去使用事件(比如没有这种信号然我们去处理这个动作),此时我们就要去重写事件处理函数去去实现特定操作。

事件的处理

在Qt,用QEvent表示事件,继承QEvent的有许多事件,如下图:
​​​​​​​​​​​​​​事件继承关系
用户进行某些操作如点击,拖拽,键盘,事件变化就会调用产生的事件处理函数。
信号槽是通过connect绑定对应的事件和信号,但是事件这里是通过重写继承父类的虚函数也就是事件函数来实现的。
我们以鼠标进入与鼠标离去这两个事件为例:
在这里插入图片描述

标签事件

第一种方法我们使用designer创建了一个label,之后我们创建了一个label类并继承QLabel,此时我们声明并重写了虚函数,在ui界面中右键提升当前的标签为我们自己的label,此时运行观察情况发现可以进行事件的触发了。

在头文件中以声明
//重写虚函数
void Label::enterEvent(QEnterEvent *event)
{qDebug()<<"调用进入事件";(void)event;
}void Label::leaveEvent(QEvent *event)
{(void)event;qDebug()<<"调用离开事件";
}

第二种方法就是直接编码,创建我我们这个label并且进行设置运行。

//在构造中进行标签的创建。Label * label=new Label(this);QRect rect=label->geometry();label->setGeometry(rect.x(),rect.y(),100,100);label->setStyleSheet("QLabel {color:blue;}");label->setStyleSheet("QLabel { border: 1px solid black; }");//设置边框label->setText("这是一个标签");

当我们鼠标进入标签区域或者离开就会掉用
在这里插入图片描述

鼠标事件

接下来我们在看看点击事件:
创建一个button为了能重写它的虚函数,老样子我们还是创建一个button类,在里面进行重写,并提升ui中的button。

void button::mousePressEvent(QMouseEvent *event)
{(void) event;if(event->button()==Qt::LeftButton){qDebug()<<"左键";}else if(){qDebug()<<"右键";}qDebug()<<"用户点击"<<event->x()<<event->y();
}
void button::mouseReleaseEvent(QMouseEvent *event)
{qDebug()<<"用户释放按钮";
}

除此之外,还有双击事件:

void button::mouseDoubleClickEvent(QMouseEvent *event)
{qDebug()<<"用户双击"<<event->x()<<event->y();if(event->button()==Qt::LeftButton){qDebug()<<"左键双击";}else{qDebug()<<"右键双击";}
}

Alt
鼠标也存在移动事件,但是直接写你会法案并不会触发,这是因为鼠标移动不同于点击,这个时间很频繁,移动一下,就会产生大量的移动事件,为了保证程序流畅,Qt默认不会对数表进行追踪,除非你真的需要该功能。

//对需要定位鼠标位置的组件进行设置this->setMouseTracking(true);
void Widget::mouseMoveEvent(QMouseEvent *event)
{qDebug()<<"鼠标移动"<<event->x()<<","<<event->y();
}

在这里插入图片描述

滚轮事件

在这里插入图片描述

void Widget::wheelEvent(QWheelEvent *event)
{qDebug()<<"滚轮滑动"<<event->angleDelta();
}

在这里插入图片描述
往上滑就是(0,120),往下滑就是(0,-120)。

按键事件

对于按键事件,在我们之前学的某些组件(如按钮)就提供有快捷键的设置,它的设计就是按键事件家信号槽实现的。
在这里插入图片描述

//按下某个字母,显示它的枚举值
void Widget::keyPressEvent(QKeyEvent *event)
{int content=event->key();//qt将所有按键都枚举为一个整型QRect rect=label->geometry();label->setGeometry(rect.x(),rect.y(),100,100);label->setStyleSheet("QLabel {color:blue;}");label->setStyleSheet("QLabel { border: 1px solid black; }");//设置边框label->setText(QString::number(content));
}
//需要注意的一点是,我们的按键触发是要对应的窗口在激活状态,而不是后台
    if((event->key()==Qt::Key_A) && (event->modifiers()==Qt::ControlModifier)){qDebug()<<"按下了组合键 a+ctrl";}

定时器事件

我们之前就了解过有一个组件就叫定时器QTimer,它的内部实现就是基于一个事件QTimerEvent.QObject也提供了一个函数timerEvent周期性触发一些操作。当然该接口还要基于两个函数使用:
QStartTimer与QKillTimer—打开定时器与关闭定时器。这里我们在designer中创建一个LCDnumber配合我们定时器使用。⭐️
在这里插入图片描述

//构造函数中//开启定时器并指定周期Timerid=this->startTimer(1000);//每过一秒就触发定时器事件ui->lcdNumber->display(10);//设置初始值
//事件处理    
void Widget::timerEvent(QTimerEvent *event)
{//先判断是哪个定时器触发的if(event->timerId()!=this->Timerid){//不是就忽略}else{//获取lcdNumber的事件int val=ui->lcdNumber->value();if(val<=0){//关闭定时器,参数为timerid,一个定时器的表标识this->killTimer(this->Timerid);return;}val-=1;//每次过一秒-1ui->lcdNumber->display(val);//在设置给定时器,表示每过一秒}
}

可以看到我们自己去搞了一个定时器,通过lcdnumber显示出来。不过实际中,我们还是直接hi用QTimer这个类型进行定时器的设置即可,它内部有定时器处理。我们可以对比看看

  //设置LCDNumber
ui->lcdNumber->display(10);//设置初始值
time=new QTimer(this);//创建timea对象
connect(time,&QTimer::timeout,this,&Widget::handle);
time->start(1000);//参数出发定时器为周期,单位msint num=ui->lcdNumber->intValue();if(num<=0){this->time->stop();return ;}ui->lcdNumber->display(num-1);

窗口事件

​窗口事件主要有两个窗口移动事件moveEvent,以及窗口ReiszeEvent 窗口大小改变触发事件。⭐️
在这里插入图片描述
在这里插入图片描述

void Widget::moveEvent(QMoveEvent *event)
{qDebug()<<"窗口位置发生改变"<<event->pos();
}void Widget::resizeEvent(QResizeEvent *event)
{qDebug()<<"窗口大小改变"<<event->size();
}

事件派发器

事件分发器
概述
在 Qt 中,事件分发器(Event Dispatcher) 是⼀个核⼼概念,⽤于处理 GUI 应⽤程序中的事件。事件分发器负责将事件从⼀个对象传递到另⼀个对象,直到事件被处理或被取消。每个继承⾃ QObject类 或QObject类 本⾝都可以在本类中重写 bool event(QEvent *e) 函数,来实现相关事件的捕获和拦截。
事件分发器⼯作原理
在 Qt 中,我们发送的事件都是传给了 QObject 对象,更具体点是传给了 QObject 对象的 event() 函数。所有的事件都会进⼊到这个函数⾥⾯,那么我们处理事件就要重写这个 event() 函数。event() 函数本⾝不会去处理事件,⽽是根据 事件类型(type值)调⽤不同的事件处理函数。事件分发器就是⼯作在应⽤程序向下分发事件的过程中。

2.Qt文件操作

早在c语言/c++我们就知晓了基本的文件操作,c语言我们可以通过fopen,fread,fwrite,fclose进行文件打开与读写,在c++中我们通过ofsream.ifstream进行文件打开关闭,使用<< >>进行读写,并且在linux中我们也学了系统接口,open,read,write,read。进行文件读写与打开关闭。而对于Qt,Qt自身也封装了文件的接口方便我们读写,这些操作都是QFile类里的成员。💖
在这里插入图片描述
QFile的继承也是分了几层,首先是最底层的IO装置类,这个的类里面的操作就是底层对磁盘的操作,之后又有6大类继承子该类,分别有文件类(作用于我们文件的读写),套接字类(QT对网络套接字也进行了封装,方便我们更简单的进行网络通信),串口类(比较古老的一个类。用于嵌入式通信),蓝牙套接字(顾名思义,用于蓝牙通信的网络编程),QProcess(进程类,可以让我们进行进程操作,如进程替换),最后是缓冲区类(辅助文件操作)。
而对于QFileDevice,有两个子类,QFile就是我们要来读写文件的类,QSaveFile是一个特殊的文件类,应用场景-在我们对旧文件修改,删除内容后编写新文件是出错了,此时也看不到旧文件。但是savefile文件读写会在提供一个缓冲文件,QTempararyFile,继承QFile表示一个临时文件类。

QFile的基本使用

文件打开方式 分别是:未打开,只读,只写。追加,打开清除,文本方式结尾用\n,无缓冲区的方式打开,不存在就创建的方式打开

void Widget::on_pushButton_open_clicked()
{//弹出文件框,用户选择打开哪个文件QString path=QFileDialog::getOpenFileName(this);//返回路径//构造QFile对象QFile file(path);//只读方式打开文件if(!file.open(QIODevice::ReadOnly  )){qDebug()<<path<<":打开失败";return ;}//读取文件QString content=file.readAll();//读取文件所有内容  ffile.readLine();//读取一行//如果是二进制文件不能用QString//关闭文件file.close();//将读到的内容设置到输入框ui->textEdit->setText(content);}void Widget::on_pushButton_save_clicked()
{//弹出保存到的位置的对话框并返回选择的路径QString path=QFileDialog::getSaveFileName(this);//返回路径//根据用户选择的路径构造QFile对象QFile file(path);//以写的方式打开文件if(!file.open(QIODevice::WriteOnly)){qDebug()<<path<<":打开失败";return ;}//将内容全写到文件中const QString content=ui->textEdit->toPlainText();file.write(content.toUtf8());//在写的时候转化格式//关闭文件file.close();
}

当然文件操作不仅仅是读文件写文件,还有其它操作,QFileInfo 是 Qt 提供的⼀个⽤于获取⽂件和⽬录信息的类,如获取⽂件名、⽂件⼤⼩、⽂件修改⽇期等。QFileInfo类中提供了很多的⽅法,常⽤的有如下

isDir() 检查该⽂件是否是⽬录;
• isExecutable() 检查该⽂件是否是可执⾏⽂件;
• fileName() 获得⽂件名;
• completeBaseName() 获取完整的⽂件名;
• suffix() 获取⽂件后缀名;
• completeSuffix() 获取完整的⽂件后缀;
• size() 获取⽂件⼤⼩;
• isFile() 判断是否为⽂件;
• fileTime() 获取⽂件创建时间、修改时间、最近访问时间等;

     QString path=QFileDialog::getOpenFileName(this);//返回路径//根据路径构造QFileinfo对象QFileInfo info(path);qDebug()<<info.fileName();//打印文件名qDebug()<<info.suffix();//打印文件后缀qDebug()<<info.path();//打印文件路径qDebug()<<info.size();//打印文件大小qDebug()<<info.isFile();//是否为普通文件qDebug()<<info.birthTime().toString();//创建时间qDebug()<<info.lastModified().toString();//罪行修改时间qDebug()<<info.lastRead();//最新读的时间

3.Qt多线程

Qt的多线程和linux的多线程本质上是一个东西,Qt对系统的线程进行了封装,这使得即使是windows或者linux一套代码都适用,在Qt中使用的是QHread类,重写类中的run函数-程序入口函数。
但是这种写法很多c++大佬认为,由于堕胎有一个去查虚函数表的过程,者在运行过程中带来了一定的时间消耗,他们会更多的使用回调的方法。
但是对于客户端开发来说,其实这点性能可有可无,QT也并不太追求。😎

run()线程的⼊⼝函数
start()通过调⽤ run() 开始执⾏线程。操作系统将根据优先级参数调度线程。如果线程已经在运⾏,这个函数什么也不做
currentThread()返回⼀个指向管理当前执⾏线程的 QThread的指针。
isRunning()如果线程正在运⾏则返回true;否则返回false。
sleep() / msleep() /usleep()使线程休眠,单位为秒 / 毫秒 / 微秒
wait()阻塞线程,直到满⾜以下任何⼀个条件:与此 QThread 对象关联的线程已经完成执⾏(即当它从run()返回时)。如果线程已经完成,这个函数将返回 true。如果线程尚未启动,它也返true,已经过了⼏毫秒。如果时间是 ULONG_MAX(默认值),那么等待永远不会超时(线程必须从run()返回)。如果等待超时,此函数将返回 false。这提供了与 POSIX pthread_join() 函数类似的功能。
terminate()终⽌线程的执⾏。线程可以⽴即终⽌,也可以不⽴即终⽌,这取决于操作系统的调度策略。在terminate() 之后使⽤ QThread::wait() 来确保。
finished()当线程结束时会发出该信号,可以通过该信号来实现线程的清理⼯作。

使用线程

这里写一个例子,创建一个线程来进行计时器来计时。首先创建一个新类,然后去继承QThread,之后去重写run方法即可。
在继承的线程类中,我们重写run,进行每1s发送一个信号

void Thread::run()
{//当然创建了线程,也无法在线程里修改主窗口//这里我们可以创建一个定时器,每次循环减一秒for(int i=0;i<10;i++){sleep(1);//成员函数 --休眠一秒emit notified();}}

在主界面我们进行信号的槽的连接,以及接收信号,只要接收信号就去执行:

//构造函数中//链接信号槽connect(&_thread,&Thread::notified,this,&Widget::handle);//启动线程_thread.start();//该信号槽每过一秒就会接受送信号执行:
void Widget::handle()
{//次数刷新界面中lcdnumber的值int val=ui->lcdNumber->value();val--;ui->lcdNumber->display(val);
}

注意在创建类继承之后重构一下项目,否则可能会报错或者没语法提示。通过以上这种方式我们简单的认识了QThread.与服务端不同,服务端主要是利用多线程高效的的处理并发请求,而客户端更加关注的是用户体验,将一些消耗的的代码使用线程来执行,以减少等待时间,例如一些密集的IO操作文件读写等。

线程锁

使用多线程就不得不提到线程安全了,常用的方法就是线程锁了,这里写一个简单的同步例子-让两个线程同时去执行一个循环加一加到5000,第一次没加锁前,结果小于10000,第二次枷锁了后结果正确。😁

void Thread::run()
{for(int i=0;i<5000;i++){//串行化的进行++,而不是并发加加,这可能会导致结果异常mutex.lock();num++;mutex.unlock();}
}
//创建两个线程对象Thread thread1;Thread thread2;thread1.start();thread2.start();//增加线程等待thread1.wait();thread2.wait();//估计是使用了条件变量进行等待,使主线程等待这两个线程运行完qDebug()<<Thread::num;

对于线程锁,我们在开发过程中为了确保他得到了释放,会使用智能指针来管理锁,让锁在出了作用域的时候自动销毁,C++也提供了智能指针锁,lokc_guard guard(mutex).那么Qt也有换了个名字叫做QMutexLocker,使用该方式管理锁,我们不用进行管理解锁。

void Thread::run()
{for(int i=0;i<5000;i++){//串行化的进行++,而不是并发加加,这可能会导致结果异常QMutexLocker mutexlocker(&mutex);// mutex.lock();num++;//mutex.unlock();}
}

既然有生产消费者模型,那肯定哈有读者模型,Qt也对这种模型的特殊锁进行了封装:
QReadWriteLocker、QReadLocker、QWriteLocker
特点:
QReadWriteLock 是读写锁类,⽤于控制读和写的并发访问。
QReadLocker ⽤于读操作上锁,允许多个线程同时读取共享资源。
QWriteLocker ⽤于写操作上锁,只允许⼀个线程写⼊共享资源。
⽤途:在某些情况下,多个线程可以同时读取共享数据,但只有⼀个线程能够进⾏写操作。读写锁提供了更⾼效的并发访问⽅式。用法与上述智能指针锁一样

connect的第五个参数

1、线程函数内部不允许操作 UI 图形界⾯,⼀般⽤数据处理;
2、connect() 函数第五个参数表⽰的为连接的⽅式,且只有在多线程的时候才意义。connect() 函数第五个参数为 Qt::ConnectionType,⽤于指定信号和槽的连接类型。同时影响信号的传递⽅式和槽函数的执⾏顺序。Qt::ConnectionType 提供了以下五种⽅式:

Qt::AutoConnection在 Qt 中,会根据信号和槽函数所在的线程⾃动选择连接类型。如果信号和槽函数在同⼀线程中,那么使⽤ Qt:DirectConnection 类型;如果它们位于不同的线程中,那么使⽤Qt::QueuedConnection 类型。
Qt::DirectConnection当信号发出时,槽函数会⽴即在同⼀线程中执⾏。这种连接类型适⽤于信号和槽函数在同⼀线程中的情况,可以实现直接的函数调⽤,但需要注意线程安全性。
Qt::DirectConnection当信号发出时,槽函数会⽴即在同⼀线程中执⾏。这种连接类型适⽤于信号和槽函数在同⼀线程中的情况,可以实现直接的函数调⽤,但需要注意线程安全性。
Qt::QueuedConnection当信号发出时,槽函数会被插⼊到接收对象所属的线程的事件队列中,等待下⼀次事件循环时执⾏。这种连接类型适⽤于信号和槽函数在不同线程中的情况,可以确保线程安全。
Qt::BlockingQueuedConnection与 Qt:QueuedConnection 类似,但是发送信号的线程会被阻塞,直到槽函数执⾏完毕,这种连接类型适⽤于需要等待槽函数执⾏完毕再继续的场景,但需要注意可能引起线程死锁的⻛险。
Qt::UniqueConnection这是⼀个标志,可以使⽤位或与上述任何⼀种连接类型组合使⽤。

条件变量和信号量

既然有了锁,那么条件变量也不能少,经典的生产者消费者模型就是基于锁与条件变量的,除了条件变量和锁能面对这种出场景下,灵活的使用信号量则能大大减少负担,效果和条件变量一致。当然linux学习的条件变量与信号量这里都与QT这里的一致,本质也是基于系统封装的。
对于QT中条件变量封装为一个类–QWaitCondition.提供了两个方法wait进行线程等待,wake进行唤醒,还有一个wakeall(唤醒所有)这个两个唤醒作用一致。在这里当然也存在之前学的Linux时的多线程下生产消费模型的虚假唤醒情况,不知大家是否还记得清呢----使用循环进行条件的判断。
对与QT中的信号量也是封装成一个类,叫做QSemaphore,创建信号量对象的时候需要设定初始值QSemaphore semaphore(2).😃

QMutex mutex;
QWaitCondition condition;
//在等待线程中
mutex.lock();
//检查条件是否满⾜,若不满⾜则等待
while (!conditionFullfilled()) //while防止虚假唤醒
{condition.wait(&mutex); //等待条件满⾜并释放锁
}
//条件满⾜后继续执⾏
//...
mutex.unlock();
//在改变条件的线程中
mutex.lock();
//改变条件
changeCondition();
condition.wakeAll(); //唤醒等待的线程
mutex.unlock();

😍 信号量的使用

QSemaphore semaphore(2); //同时允许两个线程访问共享资源
//在需要访问共享资源的线程中
semaphore.acquire(); //尝试获取信号量,若已满则阻塞
//访问共享资源
//...
semaphore.release(); //释放信号量
//在另⼀个线程中进⾏类似操作

这里都不做详细介绍,大家感兴趣只需要对linux的线程掌握熟透,其他的封装的也都会用了。😭

4.Qt网络编程

😇💘看到这里,我们就来到了网络编程,还是在与linux中学习到soketAPI的一样,只不过Qt对其进行封装。其次,虽然早都有了socket编程,但对于c++,至今都没有封装网络编程的API。由于我们的网络编程主要就是应用层代码的编写,而应用层需要传输层的支持,对于传输层主要就两种通信方式TCP与UDP,由于这两个之间的通信方式差别很大,因此QT提供了两套网络编程的API。
首先在进行QT网络编程时,需要在。proc文件中添加ntework模块:在这里插入图片描述
对于这个.proc文件,其实是QT工程的pro文件,在创建工程时由QTCreater自动创建,我们可以往里面添加内容,增加库文件的声明,包含路径、预处理器定义,生成目录,输出中间目录等等设置,通过这样的方式,在项目创建生成时会快速的生成基本的使用环境,二在你需要使用其他模块时,在修改.proc文件,主要也是因为要对一个项目更好的管理各种库与模块。对这些库,QT也提供了静态库与动态库两种方式。❤️

UDP Socket

在Qt中,对于UDP Socket提供了两个类,QUdpSocket与QNetworkDategram,QUdpSocke就是udp的socketAPI文件。如下:

名称类型说明对应的原生API
bind(const QHostAddress&,quint16)⽅法绑定指定的端⼝号.bind
receiveDatagram()⽅法返回 QNetworkDatagram . 读取⼀个 UDP 数据报recvfrom
writeDatagram(constQNetworkDatagram&)⽅法发送⼀个 UDP 数据报sendto
readyRead信号在收到数据并准备就绪后触发⽆ (类似于 IO 多路复⽤的通知机制)

QNetworkDategram代表一个udp数据包。QT的网络接口都不是阻塞式的,而是利用了信号槽机制,如果有客户按连接服务器并发送数据,即服务端收到了请求,此时就去对应的槽函数里去处理请求。对应的,客户端再发完数据,服务端也会发一个响应信号(响应信息),此时去槽函数中处理详细信息。
在这里插入图片描述
先编写服务端,如下是服务端:

#include "widget.h"
#include "ui_widget.h"
#include<QUdpSocket>
#include<QMessageBox>
#include<QNetworkDatagram>
Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);//创建套接字对象udpserver=new QUdpSocket(this); //作为父元素挂到对象树中不需手动释放了//设置窗口标题this->setWindowTitle("服务器");//服务器链接读信号槽,通过信号槽机制来驱动服务器connect(udpserver,&QUdpSocket::readyRead,this,&Widget::ProcessRequest);//绑定端口号,--注意顺寻先链接信号槽,再绑定端口号,注意收到信号就可以请求读取,不然有丢失请求bool result=udpserver->bind(QHostAddress::Any,8080); //绑定的ip地址可以侦听IPv4,ipv6if(!result){qDebug()<<"绑定失败";//弹出对话框进行提示QMessageBox::warning(this,"警告",udpserver->errorString());//这里的错误信息本质就是errnoreturn ;}}Widget::~Widget()
{delete ui;
}//请求处理,主要做三件事读取请求并解析,根据请求计算相应,把请求写回客户端
void Widget::ProcessRequest()
{//1.读取请求并解析const QNetworkDatagram& requestdatagram= udpserver->receiveDatagram();//读取之后返回请求的数据报QString request=requestdatagram.data();//返回的是一个字节数组 byteArray,调用data获取字符串//2.根据请求计算响应(由于我们这里就是单纯显示一下信息,因此无计算)const QString& response=this->procees(request);//3.把响应写回给客户端QNetworkDatagram responseDatagram= QNetworkDatagram(response.toUtf8(),requestdatagram.senderAddress(),requestdatagram.senderPort());//通过respone构造响应的数据报,从请求数据报中获取ip地址与端口号,即把响应写回给客户端udpserver->writeDatagram(responseDatagram);//打印信息到界面QString log="["+(requestdatagram).senderAddress().toString()+","+QString::number((requestdatagram).senderPort())+" ]"+"request:"+request+",response:"+response+" .";ui->listWidget->addItem(log);
}QString Widget::procees(QString &request)
{//设计一个服务器,在处理请求计算响应的过程是非常复杂的,不过我们这里不设计负责的业务场景return request;
}

如下是客户端😭

#include "widget.h"
#include "ui_widget.h"
#include<QNetworkDatagram>
//客户端来连接服务端
static const  QString serverip="192.168.1.3";//服务器地址,z这里就是自己连自己,本地环回
const quint16 serverport=8080;//服务器端口号quint6   就是  uint6_t
Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);udpclient=new  QUdpSocket(this);//发送完数据之后,还需要服务端会发送响应给客户端,通过信号槽来处理返回的数据connect(udpclient,&QUdpSocket::readyRead,this,&Widget::processResponse);
}Widget::~Widget()
{delete ui;
}void Widget::on_pushButton_clicked()
{//获取输入框数据QString content=ui->lineEdit->text();//构建请求//将QString转化为字节数组QNetworkDatagram requestdatagram(content.toUtf8(),QHostAddress(serverip),serverport);//    QNetworkDatagram(const QByteArray &data, const QHostAddress &destinationAddress = QHostAddress(), quint16 port = 0); // implicit//发送请求给服务端udpclient->writeDatagram(requestdatagram);//直接写回udpclient  ---对应的文件描述符//发送的时候把发送的请求显示到listwidget上ui->listWidget->addItem("客户端说:"+content);}void Widget::processResponse()
{//槽函数处理响应const QNetworkDatagram& UdpResponse=udpclient->receiveDatagram();//获取响应的数据报QString response=UdpResponse.data();//数据报是字节数组,data转化QString//把响应数据显示到界面中ui->listWidget->addItem("服务器说(response):"+response);
}

在这里插入图片描述
我们知道udp不可靠,面向数据报,无连接。因此大多数情况我们使用tcp进行传输的,tcp是有连接,可靠的,面向字节流的。

TCP Socket

QT中TCP Socket常用的两个类是QTcpServer和QTcpSocket,其中QTcpServer用于监听端口,获取客户端连接。如下是核心方法:

QTcpServer

名称类型说明原生API
listen(const QHostAddress&,quint16 port)⽅法绑定指定的地址和端⼝号, 并开始监听.bind 和 listen
nextPendingConnection()⽅法从系统中获取到⼀个已经建⽴好的tcp 连接.返回⼀个 QTcpSocket , 表⽰这个客⼾端的连接.通过这个 socket 对象完成和客⼾端之间的通信.accept
newConnection信号有新的客⼾端建⽴连接好之后触发.⽆ (但是类似于 IO 多路复⽤中的通知机制)

我们知道在套接字层面,tcp比udp多了两个步骤,一个是绑定知道后要监听套接字,将主动套接字转化为监听套接字,此时就可以接受来自客户端的请求了,具体来说,监听套接字,是服务器作为客户端连接请求的一个端点,它被创建一次,并存在于服务器的整个生命周期。已连接套接字是客户端与服务器之间已经建立起来了的连接的一个端点,服务器每次接受连接请求时都会创建一次已连接套接字,它只存在于服务器为一个客户端服务的过程中
其二也就是accept,告诉服务端三次握手已经成功,建立连接了,可以进行传输
这里使用应该是将newconnection与nextpendingconnection结合使用,与我们linux学到的一样,在监听之后,如果请求建立连接,此时再去accept–数据传输。当然这里的默认是阻塞式的传输,我们还可以对标非阻塞式的epoll模型进行非阻塞式传输。

QTcpSocket

QTcpSocket这个类用于客户端与服务端之间的数据交互:

名称类型说明原生API
readAll()⽅法读取当前接收缓冲区中的所有数据返回 QByteArray 对象…read
write(const QByteArray& )⽅法数据写⼊ socket 中.write
deleteLater⽅法暂时把 socket 对象标记为⽆效. Qt会在下个事件循环中析构释放该对象.⽆ (但是类似于 “半⾃动化的垃圾回收”)
readyRead信号有数据到达并准备就绪时触发.⽆ (但是类似于 IO 多路复⽤中的通知机制)
disconnected信号连接断开时触发⽆ (但是类似于 IO 多路复⽤中的通知机制)

编写服务端-这里的用例我们跟udp的一样,服务端使用一个listwidget显示收到的请求以及做出的响应:

#include "widget.h"
#include<QMessageBox>
#include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);this->setWindowTitle("TcpServer");tcpserver=new QTcpServer(this);//创建用来进行通信的socket对象//信号槽的绑定connect(tcpserver,&QTcpServer::newConnection,this,&Widget::ProcessConnection);//绑定并监听,先准备好信号到来的准备工作---信号槽的编写,请求的处理//最后一步才是绑定bool result=tcpserver->listen(QHostAddress::Any,8080);if(!result){QMessageBox::warning(this,"警告-绑定失败",tcpserver->errorString());exit(1);}
}Widget::~Widget()
{delete ui;
}void Widget::ProcessConnection()
{//获取连接之后的scoket对象,即对应accept之后连接客户端的端口QTcpSocket*  clientSocket=tcpserver->nextPendingConnection();//向窗口打印日志,某某客户端上线QString log="[ "+clientSocket->peerAddress().toString()+QString::number(clientSocket->peerPort())+" ],客户端上线";//这里的peer指的是对等端//这里的对端指的就是所连接的客户端的端口ui->listWidget->addItem(log);//连接建立完毕,此时就要去看是否有收到请求的信号,如果有,就去处理请求,对象为accept之后的对等的端口号connect(clientSocket,&QTcpSocket::readyRead,this,[=](){//获取请求QString request=clientSocket->readAll();//此时的数据流就是字节流//根据请求处理响应,当前无业务需求直接返回QString response=process(request);//获取到了响应//获取到之后写会给客户端clientSocket->write(response.toUtf8());//转为utf8,就可获取到字节数组//写回的同时,记录在窗口上QString log1="[ "+clientSocket->peerAddress().toString()+QString::number(clientSocket->peerPort())+" ]"+",response:"+response;ui->listWidget->addItem(log1);});//上述代码不够严谨,对于回显服务器足够了,但是我们并不能保证在传输过来的请求是不是完整的请求,还是分段的//更严谨的做法,是提供一块缓冲区,将请求都放在一起,在进行一条条完整的请求的解析------协议的定制//完成以上工作,之后如果客户端断开连接也需要处理,通过信号槽获取断开联机的信号之后处理connect(clientSocket,&QTcpSocket::disconnected,this,[=](){//打印断开连接的日志QString log="[ "+clientSocket->peerAddress().toString()+QString::number(clientSocket->peerPort())+" ]"+"客户端请求断开连接";ui->listWidget->addItem(log);//手动释放socket,这里会为每个客户端维护一个accept之后的socket用于通信,所以如果没释放,将会造成严重的内存泄漏与文件描述符泄露//delete clientSocket;clientSocket->deleteLater();//释放需要考虑两个问题:1.释放一定是在最会一步     2.释放一定会被执行不会被异常,retrurn 跳过//其次除了直接delete,还有一个接口deletelater  ,表示在下一层循环(信号槽处于一个循环)进行释放,通过这种方式,上述的两个问题都不需要考虑了。});}QString Widget::process(QString &request)
{return request;
}

与编写服务端不同,服务端需要两个类,QTcpServer用来建立连接和销毁链接,而QTcpSocket是用来处理请求的,对于客户端来说,只需要发送请求和处理响应,因此不需要Tcpserver.

#include "widget.h"
#include "ui_widget.h"
#include<QMessageBox>
Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);this->setWindowTitle("客户端");clientSocket=new QTcpSocket(this);//向服务器发起连接clientSocket->connectToHost("127.0.0.1",8080);//建立三次握手,该接口是非阻塞式建立连接//连接建立成功之后便可以发送请求//请求发送完,客户端就会收到服务端的响应,因此此时还需要处理响应connect(clientSocket,&QTcpSocket::readyRead,this,[=](){//处理响应,先获取响应QString response=clientSocket->readAll();//相应内容显示到界面上ui->listWidget->addItem("服务器响应: "+response);});//等待连接建立的结果,确认是否建立成功bool ret=clientSocket->waitForConnected();//通过该接口,也使得socket阻塞式的等待if(!ret){//连接建立失败QMessageBox::warning(this,"警告,连接失败",clientSocket->errorString());exit(1);}//连接建立成功之后便可以发送请求//请求发送完,客户端就会收到服务端的响应,因此此时还需要处理相应,读取响应}Widget::~Widget()
{delete ui;
}//点击按钮就是发送请求
void Widget::on_pushButton_clicked()
{//内容就是数据QString content=ui->lineEdit->text();//面向字节流,不需要构建请求,直接写给服务端clientSocket->write(content.toUtf8());//回显发送数据给屏幕ui->listWidget->addItem("客户端发送: "+content);//清空输入框内容ui->lineEdit->setText("");
}

😭注意事项,之前我们在linux中编写Tcp服务器时,一个客户端连接的时候是正常的,但是当多个客户端来连接就只有一个客户端生效,因此当时使用的是多线程的方式(创建了一个子线程用来处理连接之后的进行数据传输的操作),但是我们当前编写的这个,多个客户端都可以访问。首先tcp服务器这里并不是要靠多线程才能达到多台连接,至于为什么linux哪里会出现问题,这是因为在linux的时候,由于我们是通过循环来处理连接和请求的,写的是双重循环,里层循环还没结束,外层循环就不能快速的第二次调用accept处理其他请求,多线程是的两个循环独立,因此可以多个访问,而qt这里使用的是信号槽机制,很好的解决了这个问题。

HTTP的编写

相对于传输层的TCP和UDP,使用这两个通信,还需要我们自定格式再加上协议才能较好的应用层上进行通信,但是我们也可以使用现成的应用层协议–http,不需要我们自己封装,底层基于TCP实现,相比之下http的使用是更为广泛的。
由于QT主要使用看来编写客户端的,因此QT也只提供了客户端的组件,是没有服务器的库。
这里我们对QT的http客户端编写也大概做一下简单的实现。
首先http是有两种请求的,一种get请求,一种post请求,请求报文的格式我们也在linux中介绍过,请求头+请求行+(空行)+正文,对应的响应报文-响应头加响应行+响应报文。
关键类主要是三个. QNetworkAccessManager , QNetworkRequest , QNetworkReply .😭
QNetworkAccessManager

方法说明
get(const QNetworkRequest& )发起⼀个 HTTP GET 请求. 返回 QNetworkReply 对象
post(const QNetworkRequest& , constQByteArray& )发起⼀个 HTTP POST 请求. 返回 QNetworkReply 对象

这里的返回值是一个QNetworkReply对象,但不含 body(正文),该对象就表示了一个类的请求,如果想添加body,在 QNetworkAccessManager 的 post ⽅法中通过单独的参数来传⼊ body。
QNetworkRequest

方法说明
QNetworkRequest(const QUrl& )通过 URL 构造⼀个 HTTP 请求
setHeader(QNetworkRequest::KnownHeadersheader,const QVariant &value)设置请求头

其中的 QNetworkRequest::KnownHeaders 是⼀个枚举类型, 常⽤取值:

方法说明
ContentTypeHeader描述 body 的类型.
ContentLengthHeader描述 body 的⻓度.
LocationHeader⽤于重定向报⽂中指定重定向地址. (响应中使⽤, 请求⽤不到)
CookieHeader设置 cookie
UserAgentHeader设置 User-Agen

QNetworkReply 表⽰⼀个 HTTP 响应. 这个类同时也是 QIODevice 的⼦类.

方法说明
error()获取出错状态.
errorString()获取出错原因的⽂本.
readAll()读取响应 body
header(QNetworkRequest::KnownHeadersheader)读取响应指定的header的值

接下来简单的编写一个客户端来看看,首先还是弄一个和之前tcp一样的窗口,注意这里的texEdit与plainEdit文本不一样,textEidt是在顶html会对内容进行渲染,而plianEdit则是纯文本输入框。

我们这里就简单发送一个的访问一下百度主页url
在这里插入图片描述

5.QT多媒体播放

无论是音频播放,还是视频播放,都需要在proc文件中引入新的库:

QT       +=multimedia
QT       +=multimediawidgets

音频播放

在 Qt 中,⾳频主要是通过 QSound 类来实现。但是需要注意的是 QSound 类只⽀持播放 wav 格式的⾳频⽂件。也就是说如果想要添加⾳频效果,那么⾸先需要将 ⾮wav格式 的⾳频⽂件转换为 wav 格式。这个类的用法也很简单,主要通过方法play播放声音:
注意:qt6中已经没有该类,而是使用QSoundEffect.

#include <QSoundEffect>
QSoundEffect* startSound = new QSoundEffect(this);
startSound->setSource(QUrl::fromLocalFile(":/music.wav"));
startSound->play();

以上方式是比较简单的一种方式,除此之外还有另一种方式,支持ogg,mp3格式的音频播放:
先包含这两个库

QT       +=multimedia
#include <QMediaPlayer>
#include <QUrl>
#include<QVideoWidget>
QMediaPlayer *music;
QAudioOutput *output;//头文件申明//设置播放//创建音频对象music=new QMediaPlayer(this);//播放器用来播放音乐music->setSource(QUrl("qrc:/kun.ogg"));//设置资源路径output=new QAudioOutput(this);//用来控制音乐输出music->setAudioOutput(output);//将输出设置进播放器void Widget::on_pushButton_clicked()
{QIcon start(QPixmap(":/start.png"));//音乐播放图标QIcon stop(QPixmap(":/stop1.png"));//音乐暂停图标if(music->playbackState()==QMediaPlayer::PausedState  ||music->playbackState()==QMediaPlayer::StoppedState) //判断是否为暂停或者是终止状态{music->play();//播放ui->pushButton_godess->setIcon(start);ui->pushButton_godess->setIconSize(QSize(50,50));}else if(music->playbackState()==QMediaPlayer::PlayingState){music->pause();ui->pushButton_godess->setIcon(stop);}
}

感觉这种方式貌似更好一点,这是只播放一个音乐的写法,如果是多个播放的媒体,可以使用QMediaPlayList将需要播放的媒体添加到list中:

int main(int argc, char *argv[])  
{  QCoreApplication a(argc, argv);  // 创建一个 QMediaPlayer 实例  QMediaPlayer *player = new QMediaPlayer;  // 创建一个 QMediaPlaylist 实例  QMediaPlaylist *playlist = new QMediaPlaylist(player);  // 将媒体添加到播放列表中  playlist->addMedia(QUrl::fromLocalFile("/path/to/your/audiofile1.mp3"));  playlist->addMedia(QUrl::fromLocalFile("/path/to/your/audiofile2.wav"));  playlist->addMedia(QUrl::fromLocalFile("/path/to/your/audiofile3.ogg"));  // 设置 QMediaPlayer 使用播放列表  player->setPlaylist(playlist);  // 连接信号和槽以处理播放状态变化  QObject::connect(player, &QMediaPlayer::mediaStatusChanged,  [](QMediaPlayer::MediaStatus status) {  qDebug() << "Media status changed to" << status;  if (status == QMediaPlayer::InvalidMedia) {  qDebug() << "Invalid media source!";  }  });  // 开始播放播放列表中的第一个媒体  player->play();  // 应用程序将一直运行,直到你停止它  return a.exec();  
}  

视频播放

音乐和视频播放基本一致,不同的是除了音乐要设置媒体播放器,设置输出设备,视屏播放器多加了一个QVideoWidget来显示视频播放的窗口。😄

 ui->setupUi(this);player=new QMediaPlayer(this);widgetplayer=new QVideoWidget(this);output=new QAudioOutput();//设置媒体播放output->setVolume(0.5);//设置声音player->setAudioOutput(output);//设置播放画面的最小窗口this->widgetplayer->setMinimumSize(400,400);//大小默认为窗口的大小this->widgetplayer->setGeometry(0,0,400,200);//设置媒体源,就是选择的文件this->player->setSource(QUrl("qrc:/test.mp4"));//输出视频画面this->player->setVideoOutput(this->widgetplayer);this->player->play();

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/26200.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【CTF MISC】XCTF GFSJ0766 something_in_image Writeup(字符串搜索)

something_in_image 暂无 解法 用 binwalk 扫描。 binwalk badimages 找到一个 ext3 文件系统。 strings badimages | grep {*}找到 flag。 Flag Flag{yc4pl0fvjs2k1t7T}声明 本博客上发布的所有关于网络攻防技术的文章&#xff0c;仅用于教育和研究目的。所有涉及到的实验…

收音机的原理笔记

1. 收音机原理 有线广播&#xff1a;我们听到的声音是通过空气振动进行传播&#xff0c;因此可以通过麦克风&#xff08;话筒&#xff09;将这种机械振动转换为电信号&#xff0c;传到远处&#xff0c;再重新通过扬声器&#xff08;喇叭&#xff09;转换为机械振动&#xff0c…

打造精细化运维新玩法(四)

二、SLO健康度——从0到1构建SLO 和大多互联网企业一样&#xff0c;蚂蚁的基础设施侧存在众多的异构系统&#xff0c;被上层的业务应用和服务所依赖。考虑到不同系统的技术栈、架构、部署等因素&#xff0c;我们需要找到一种通用的、泛化性强的数字化方案指导和构建基础设施域内…

智能楼宇可视化:赋能智慧园区管理

图扑智慧园区楼宇可视化系统&#xff0c;集成多种数据源&#xff0c;实现全方位监控与管理&#xff0c;提升园区的资源利用率和用户满意度&#xff0c;推动智能化管理进程。

电线电缆单根燃烧试验 电缆垂直燃烧试验

电线电缆单根燃烧试验 电线电缆单根燃烧试验是一种用来评估电线电缆在受到火焰作用时的燃烧性能的测试方法。这种试验通常是将电线电缆垂直固定&#xff0c;然后使用特定的火焰源对其进行燃烧&#xff0c;以观察电线电缆的燃烧行为和燃烧速度。通过这个试验&#xff0c;可以评估…

护眼台灯哪个品牌好?几款性价比最高的护眼台灯推荐

在过去&#xff0c;科技尚未发展至如今这般先进水平时&#xff0c;晚上需要照明的时候&#xff0c;我们通常只能依赖白炽灯。尽管白炽灯以其低成本和接近自然光的显色性获得了一定的青睐&#xff0c;随着时代的发展&#xff0c;现在市面上出现了更为护眼的选择——LED台灯。然而…

一文介绍暗区突围手游 游戏特色、具体玩法和独特的玩法体验

&#x1f349; CSDN 叶庭云&#xff1a;https://yetingyun.blog.csdn.net/ 《暗区突围》是一款由腾讯魔方工作室群开发的第一人称射击游戏&#xff0c;于 2022 年 7 月 13 日正式公测&#xff0c;支持 Android 和 iOS 平台。这款游戏以从虚构的暗区收集物资并安全撤离作为最终目…

OpenGauss数据库-7.用户及角色

第1关&#xff1a;创建用户 gsql -d postgres -U gaussdb -W passwd123123 CREATE USER jackson WITH PASSWORD jackson123; 第2关&#xff1a;修改用户 gsql -d postgres -U gaussdb -W passwd123123 ALTER USER jackson WITH PASSWORD Abcd123; 第3关&#xff1a;创建角色 …

【JVM】JVM 的内存区域

Java虚拟机&#xff08;JVM&#xff09;在执行Java程序时&#xff0c;将其运行时数据划分到若干不同的内存区域。这些内存区域的管理对Java应用程序的性能和稳定性有着重要影响。JVM的内存区域主要包括以下几部分&#xff1a; 方法区&#xff08;Method Area&#xff09;&#…

Qt实现简易播放器

效果如图 源码地址&#xff1a; 简易播放器: 基于Qt的简易播放器&#xff0c;底层采用VLC源码 - Gitee.com GitHub:GitHub - a-mo-xi-wei/easy-player: 基于Qt的调用VLC的API的简易播放器

QQ音乐绿钻API接口:解锁更多音乐可能性

在我们日常生活中&#xff0c;音乐是不可或缺的一部分。无论是在上班途中&#xff0c;还是在健身房锻炼时&#xff0c;我们都可以通过听音乐来放松自己。然而&#xff0c;在现如今的音乐市场中&#xff0c;有时候我们会觉得收听的歌曲有限&#xff0c;想要尝试更多不同的音乐类…

《大道平渊》· 拾贰 —— 天下大事必作于细:做好每一件小事,必然大有所成!

《平渊》 拾贰 "天下难事必作于易&#xff0c;天下大事必作于细。" 社群一位大佬最近在研究新项目, 他做事的 "方法论" 令我深受启发。 他在测试项目时, 每一步都做的非常细致&#xff1a; 整个项目的测试都被划分为一件件小事, 然后有条不紊地推进…… …

代码随想录刷题笔记-哈希表篇

文章目录 242 有效的字母异位词(easy)力扣地址题目描述题目实例解题思路代码实现 383 赎金信(easy)力扣地址题目描述题目实例解题思路代码实现 49 字母异位词分组(mid)力扣地址题目描述题目实例解题思路代码实现 438 找到字符串中所有字母异位词(mid)力扣地址题目描述题目实例解…

免费插件集-illustrator插件-Ai插件-文本属性批处理

文章目录 1.介绍2.安装3.通过窗口>扩展>知了插件4.功能解释5.总结 1.介绍 本文介绍一款免费插件&#xff0c;加强illustrator使用人员工作效率&#xff0c;进行文本属性批处理。首先从下载网址下载这款插件 https://download.csdn.net/download/m0_67316550/87890501&am…

16个免费学习Python的网站和教程(2024年最新资源)

16个免费学习Python的网站和教程&#xff08;2024年最新资源&#xff09; 博主猫头虎的技术世界 &#x1f31f; 欢迎来到猫头虎的博客 — 探索技术的无限可能&#xff01; 专栏链接&#xff1a; &#x1f517; 精选专栏&#xff1a; 《面试题大全》 — 面试准备的宝典&#xff…

认识Django框架,使用Django 2024新手创建Django项目,使用编译工具:Pycharm

Django简单介绍 Django 是一个用 Python 编写的开源 web 应用框架&#xff0c;旨在促进快速开发、维护和部署高效、可扩展的 web 应用程序。它是遵循模型-模板-视图&#xff08;MTV&#xff09;设计模式的一个高级框架&#xff0c;尽管有时也被描述为遵循MVC&#xff08;模型-…

【Three.js】知识梳理一:Three.js概述和基础知识

1.Three.js简介 Three.js是一个基于JavaScript编写的开源3D图形库&#xff0c;利用WebGL技术在网页上渲染3D图形。它提供了许多高级功能&#xff0c;如几何体、纹理、光照、阴影等&#xff0c;以便开发者能够快速地创建复杂且逼真的3D场景。同时&#xff0c;Three.js还具有很好…

使用seq2seq架构实现英译法

seq2seq介绍 模型架构&#xff1a; Seq2Seq&#xff08;Sequence-to-Sequence&#xff09;模型是一种在自然语言处理&#xff08;NLP&#xff09;中广泛应用的架构&#xff0c;其核心思想是将一个序列作为输入&#xff0c;并输出另一个序列。这种模型特别适用于机器翻译、聊天…

运用HTML+CSS+JS做一个贪吃蛇游戏

贪吃蛇游戏 前言效果图部分源码领取源码下期更新预报 前言 H5贪吃蛇大战HTML源码&#xff0c;可在本地浏览器打开访问index.html&#xff0c;或者上传到服务器或虚拟空间进行游玩&#xff01; PC版操作 鼠标点击一下之后就可以控制方向&#xff0c;按A加速 移动端操作 左侧…

【Linux】进程8——进程创建和进程终止

1.进程创建 1.1.再谈fork 在linux中fork函数时非常重要的函数&#xff0c;它从已存在进程中创建一个新进程。新进程为子进程&#xff0c;而原进程为父进程。 #include <unistd.h> pid_t fork(void);//pid_t为整形 返回值&#xff1a;子进程中的fork()返回0&#xff…