C++及QT的线程学习

news2025/1/23 22:32:49

目录

一. 线程学习

二. 学习线程当中,得到的未知。

1. 了解以下MainWindow和main的关系

2. [=]()匿名函数 有函数体,没有函数名.

3. join和detach都是用来管理线程的生命周期的,它们的区别在于线程结束和资源的回收。

4. operator()() 仿函数

5. try-catch的使用以及细节。处理异常,try块用于包含可能出错的代码。catch块用于处理try块中发生的异常。

6. C++创建多线程

1).普通函数的多线程创建方式

2).lsmbda表达式方式创建多线程

3).仿函数方式创建多线程

4).类成员函数创建多线程


一. 线程学习

首先,我们任意建一个QT工程,按下按键让数字递增。

void Widget::on_pushButton_clicked()
{
    int i = 0;
    for(;;)
    {
        ui->lcdNumber->display(QString::number(i++));
        sleep(1);
    }
}

但是我们在槽里这样写的话,移动窗口或者按下按键会报错,因为我们既要绘制窗口,又要响应窗口移动的操作,还要执行自加的逻辑,当执行到这个自加的逻辑就已经很忙了,它就没有功夫去调用显示逻辑了。我们可以加一个Debug查看一下。

void Widget::on_pushButton_clicked()
{
    int i = 0;
    for(;;)
    {
        qDebug() << i;
        ui->lcdNumber->display(QString::number(i++));
        sleep(1);
    }
}

可以发现操作台是有数在走的,但是却没有显示。

这时候我们就得采用多线程。那么我们使用join还是detach呢,因为主线程也要执行,不可能等待子线程执行,而且主线程本身就是一个循环,比如return a.exec();主进程不会退出,所以使用detach。

void Widget::on_pushButton_clicked()
{
    std::thread my_thread(&Widget::showInfo,this);
    my_thread.detach();
}

void Widget::showInfo()
{
    int i = 0;
    for(;;)
    {
        qDebug() << i;
        ui->lcdNumber->display(QString::number(i++));
        sleep(1);
    }
}

因为是在栈中定义的my_thread,所以不需要担心资源回收的问题,函数一结束,my_thread就销毁了,不用担心线程回收的问题。

这样,我们点击按钮开始计数之后就不会卡死。

当然QT中封装了一个多线程的类,叫QThread。刚刚我们写的线程是没有退出的逻辑的,所以接下来我们使用QThread。

定义一个自定义的类,继承QThread并重写run()方法,在里面写线程执行的逻辑,定义一个信号。

#include "my_thrad.h"
#include <QDebug>
My_Thrad::My_Thrad(QObject *parent) : QThread(parent)
{

}

void My_Thrad::run()
{
    int i = 0;
    for(;;)
    {
        qDebug() << i;
        emit threadSignal(i++);
        this->sleep(1);
        if (i > 10)
        {
            break;
        }
    }
}
#ifndef MY_THRAD_H
#define MY_THRAD_H

#include <QThread>
#include <QDebug>
class My_Thrad : public QThread
{
    Q_OBJECT
public:
    explicit My_Thrad(QObject *parent = nullptr);
    ~My_Thrad()//析构函数
    {
        qDebug() << "线程退出了,并回收了线程空间";
    }
protected:
    void run() override;
signals:
    void threadSignal(int val);
};

#endif // MY_THRAD_H

这里我们添加了一个析构函数,是判断新建线程能不能结束,而要保证关闭窗口的时候,线程仍然能够执行,实现一个安全可靠的退出,我们就要:

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

Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);
    my_thread = new My_Thrad(this);//this父对象就是main.cpp中的w
    connect(my_thread,&My_Thrad::threadSignal,[=](int val){
        ui->lcdNumber->display(QString::number(val));
    });
}

Widget::~Widget()
{
    delete ui;
    my_thread->exit();
    my_thread->wait();//等待线程执行完毕,然后才退出
    delete my_thread;
//   // my_thread->deleteLater();//它是依赖于某个对象,还依赖于主事件循环存在的情况下,才是有效的,所以这样会内存泄露
}

void Widget::on_pushButton_clicked()
{
    my_thread->start();//使用start间接调用的run方法
}

因此我们在~Widget()析构函数中写上述代码,实现等待线程执行完毕,才退出,并删除my_thread,输出”线程退出了,并回收了线程空间。

二. 学习线程当中,得到的未知。

1. 了解以下MainWindow和main的关系

main()函数,非窗体程序入口函数

Mainwindow函数,是窗体程序的入口函数

2. [=]()匿名函数 有函数体,没有函数名.

3. join和detach都是用来管理线程的生命周期的,它们的区别在于线程结束和资源的回收。

join函数会阻塞当前线程,直到被调用join()的线程(子线程)执行完毕并退出,在这个过程中,调用join()的线程会一直等待,直到被等待的线程退出。如果没有调用join函数。被等待的线程退出后,它的资源不会被回收,这可能会导致内存泄漏。

有时候我们不知道是否已经join()系统提供了一个joinable()来判断是否已经join()

使用detach会让线程在后台运行,这就意味着与主线程不能直接交互了,分离后的线程不能join

但使用detach时,要注意主进程运行的时间,不然可能线程还没执行完,主进程就结束了

4. operator()() 仿函数

5. try-catch的使用以及细节。处理异常,try块用于包含可能出错的代码。catch块用于处理try块中发生的异常。

try{
//可疑代码
//将异常生成对象的异常对象传递给catch块
}catch(异常){
//对异常进行处理
}finally{
} //可以没有finally

6. C++创建多线程

1).普通函数的多线程创建方式

#include <thread>
#include <iostream>
#include <unistd.h>
#include <string>
using namespace std;
void showInfo()
{
    int i = 0;
    for(;;)
    {
        cout << i++ << endl;
        sleep(10000);
    }
}

void print(const string &s)
{
    cout<<"hello thread!"<<endl;
    cout<<s<<endl;
}

int main()
{
    /*
    thread my_thread(&showInfo);
    my_thread.join();
    my_thread.detach();
    */
    cout<<"main thread begin!"<< endl;
    string s = "hello world";
    thread t(&print,s);

    //thread t(&print);

    t.join();
    cout<<"main thread end!"<<endl;
    return 0;
}

2).lsmbda表达式方式创建多线程

#include <iostream>
#include <thread>
#include <string>
using namespace std;

int main()
{
    cout<<"main begin"<<endl;
    thread t(
        [](string s)-> void{
            cout<<"hello world!"<<endl;
            cout<<s<<endl;
        },
        "abc"
        );
    t.join();
    cout<<"main end"<<endl;
    return 0;
}

3).仿函数方式创建多线程

#include <iostream>
#include <thread>
#include <string>
using namespace std;
class MyThread{
public:
    void operator()()
    {
        cout<<"Hello World!"<<endl;
    }
};

int main()
{
    cout<<"main begin"<<endl;
    MyThread mt;
    thread t(mt);

    t.join();
    cout<<"main end"<<endl;

    return 0;
}

4).类成员函数创建多线程

#include <iostream>
#include <thread>
#include <string>
using namespace std;
class MyThread{
public:
    void print(const string &s)
    {
        cout<<"Hello World!"<<endl;
        cout<<s<<endl;
    }
};

int main()
{
    cout<<"main begin"<<endl;
    MyThread mt;
    thread t(&MyThread::print,&mt,"ac") ;

    t.join();
    cout<<"main end"<<endl;

    return 0;
}

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

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

相关文章

Spark-机器学习(4)回归学习之逻辑回归

在之前的文章中&#xff0c;我们来学习我们回归中的线性回归&#xff0c;了解了它的算法&#xff0c;知道了它的用法&#xff0c;并带来了简单案例。想了解的朋友可以查看这篇文章。同时&#xff0c;希望我的文章能帮助到你&#xff0c;如果觉得我的文章写的不错&#xff0c;请…

管理集群工具之LVS

管理集群工具之LVS 集群概念 将很多机器组织在一起&#xff0c;作为一个整体对外提供服务集群在扩展性、性能方面都可以做到很灵活集群分类 负载均衡集群&#xff1a;Load Balance高可用集群&#xff1a;High Availability高性能计算&#xff1a;High Performance Computing …

OpenCV轻松入门(九)——使用第三方库imgaug自定义数据增强器

安装命令&#xff1a;pip install imgaug 代码实现&#xff1a; import cv2 import random import matplotlib.pyplot as pltfrom imgaug import augmenters as iaa # 数据增强——缩放效果 def zoom_img(img):# 获取一个1-1.3倍的线性图像处理器&#xff0c;scale参数是缩放范…

计算机视觉 | 交通信号灯状态的检测和识别

Hi&#xff0c;大家好&#xff0c;我是半亩花海。本项目旨在使用计算机视觉技术检测交通信号灯的状态&#xff0c;主要针对红色和绿色信号灯的识别。通过分析输入图像中的像素颜色信息&#xff0c;利用OpenCV库实现对信号灯状态的检测和识别。 目录 一、项目背景 二、项目功能…

项目大集成

一 keeplived 高可用 192.168.11.11nginx keeplived192.168.11.12nginx keeplived 两台均编译安装服务器 1 主服务器修改文件&#xff1a; 2 备服务器修改文本 scp keepalived.conf 192.168.11.12:/etc/keepalived/ 3 给主服务器添加虚拟ip ifconfig ens33:0 192.168…

React.js 3D开发快速入门

如果你对 3D 图形的可能性着迷&#xff0c;但发现从头开始创建 3D 模型的想法是不可能的 - 不用担心&#xff01; Three.js 是一个强大的 JavaScript 库&#xff0c;它可以帮助我们轻松地将现有的 3D 模型集成到 React 应用程序中。因此&#xff0c;在本文中&#xff0c;我将深…

DHCP服务器配置故障转移后显示红色箭头、与伙伴服务器失去联系的解决方案

一、遇到的故障现象&#xff1a; &#xff08;主DHCP服务器与备用DHCP服务器连通性正常&#xff0c;在故障转移选项卡上却显示与伙伴失去联系、伙伴关闭&#xff0c;且ipv4协议旁边显示一个红色的小箭头&#xff09;&#xff0c;正常情况下是绿色 &#xff08;一&#xff09;…

Jenkins CI/CD 持续集成专题二 Jenkins 相关问题汇总

一 问题一 pod [!] Unknown command: package 1.1 如果没有安装过cocoapods-packager&#xff0c;安装cocoapods-packager&#xff0c;sudo gem install cocoapods-packager 1.2 如果已经安装cocoapods-packager&#xff0c;还是出现上面的错误&#xff0c;有可能是pod的安…

通过创新的MoE架构插件缓解大型语言模型的世界知识遗忘问题

在人工智能领域&#xff0c;大型语言模型&#xff08;LLM&#xff09;的微调是提升模型在特定任务上性能的关键步骤。然而&#xff0c;一个挑战在于&#xff0c;当引入大量微调数据时&#xff0c;模型可能会遗忘其在预训练阶段学到的世界知识&#xff0c;这被称为“世界知识遗忘…

Valentina Studio Pro for Mac:强大的数据库管理工具

Valentina Studio Pro for Mac是一款功能全面、操作高效的数据库管理工具&#xff0c;专为Mac用户设计&#xff0c;旨在帮助用户轻松管理各种类型的数据库。 Valentina Studio Pro for Mac v13.10激活版下载 该软件拥有直观的用户界面&#xff0c;使得数据库管理变得简单直观。…

JS - 以工厂模式和原型模式方式建造对象、JS的垃级回收机制、数组的使用

创建对象的方式 使用工厂方法来建造对象 在JS中我们可以通过以下方式进行创建对象&#xff1a; var obj {name:"孙悟空",age:18,gender:"男",sayName:function(){alert(this.name);}};var obj2 {name:"猪八戒",age:28,gender:"男",…

学习ArkTS -- 状态管理

装饰器 State 在声明式UI中&#xff0c;是以状态驱动试图更新&#xff1a; 状态&#xff08;State&#xff09;&#xff1a;指驱动视图更新的数据&#xff08;被装饰器标记的变量&#xff09; 视图&#xff08;View&#xff09;&#xff1a;基于UI描述渲染得到用户界面 说明…

Next.js+React+Node系统实战,搞定SSR服务器渲染

Next.jsReactNode系统实战&#xff0c;搞定SSR服务器渲染 Next.js React Node.js 实战&#xff1a;实现服务器端渲染&#xff08;SSR&#xff09; 项目概述 在这个项目中&#xff0c;我们将探讨如何使用 Next.js、React 和 Node.js 来构建一个服务器渲染的 web 应用程序。通…

无人驾驶 自动驾驶汽车 环境感知 精准定位 决策与规划 控制与执行 高精地图与车联网V2X 深度神经网络学习 深度强化学习 Apollo

无人驾驶 百度apollo课程 1-5 百度apollo课程 6-8 七月在线 无人驾驶系列知识入门到提高 当今,自动驾驶技术已经成为整个汽车产业的最新发展方向。应用自动驾驶技术可以全面提升汽车驾驶的安全性、舒适性,满足更高层次的市场需求等。自动驾驶技术得益于人工智能技术的应用…

unity读写本地excel_2024.4.22

using System.Collections; using System.Collections.Generic; using UnityEngine; using OfficeOpenXml; using System.IO; using Excel; using System.Data; using System; /// <summary> /// https://blog.csdn.net/Xz616/article/details/128893023 /// Unity3D操作…

【入门篇】本章包括创建云项目、数据库的使用、云存储管理、云函数的基本使用、实战举例(小程序之云函数开发入门到使用发布上线实操)

云函数 云函数相当于服务器接口的概念,它并属于小程序端代码。它是以函数的形式运行后端代码来响应事件以及调用其他服务。运行环境是Node.js。 一、基创建云函数项目 打开微信开发者工具: 打开微信开发者工具,并登录你的微信开发者账号。 创建项目: 如果还没有创建项目,你…

给字符串添加加粗标签(AC自动机+Python)

可以暴力解决&#xff0c;但是为了锻炼一下ac自动机的编程&#xff0c;我们使用ac自动机。 ac自动机主要维护两个列表&#xff0c;一个列表ch&#xff0c;ch[f][idx]表示从父节点f向idx这个方向走&#xff0c;走到的节点。另一个列表nex&#xff0c;nex[i]表示节点i回跳边的节…

机器学习 -- 分类问题

场景 探讨了一个回归任务——预测住房价格&#xff0c;用到了线性回归、决策树以及随机森林等各种算法。本次中我们将把注意力转向分类系统。我们曾经对MNIST进行了分类任务&#xff0c;这次我们重新回到这里&#xff0c;细致的再来一次。 开始 获取数据 Scikit-Learn提供了…

BFS解决FloodFill算法:(Leetcode:200. 岛屿数量)

题目链接&#xff1a;200. 岛屿数量 - 力扣&#xff08;LeetCode&#xff09; 本题由于没有给出开始搜索的位置&#xff0c;所以每一个位置都要进行一次广度优先搜索 另外为了不修改原数组数据&#xff0c;需要设置一个bool类型的二维数组vis来判断某个位置是否被搜索过 cl…

负载均衡的原理及算法

一、定义 负载均衡&#xff08;Load Balancing&#xff09;是一种计算机网络和服务器管理技术&#xff0c;旨在分配网络流量、请求或工作负载到多个服务器或资源&#xff0c;以确保这些服务器能够高效、均匀地处理负载&#xff0c;并且能够提供更高的性能、可用性和可扩展性。…