深入解析 Go 语言 GMP 模型:并发编程的核心机制

news2025/1/12 18:49:41

前言

本章是Go并发编程的起始篇章,在未来几篇文章中我们会围绕Go并发编程进行理论和实战的学习,欢迎关注我哦!

本章主要以介绍GMP模型为主,偏向于面试和八股,目的是让小伙伴们注重于知识本身,面向面试,面向八股,面向加薪。

Go语言自诞生以来,就以其简洁、高效的并发模型著称。而这其中的核心正是GMP模型。理解GMP模型的演进历程,能帮助我们更好地掌握Go的并发编程。而Goroutine作为Go中的核心概念,极大地简化了并发编程的复杂度。本文将详细阐述Go语言GMP模型的演变过程,并深入解析其设计理念和优点,并详细介绍Goroutine的基本概念、优势及其使用方法,并结合具体代码示例进行说明。

面试题目

在阅读本文前,先带着以下几个关于GMP模型的面试题目进行思考,以加深理解和掌握:

  1. 什么是GMP模型?请解释其基本概念。
  • 回答要点:解释G、M、P的概念及其在调度模型中的角色。
  1. 如何理解GMP模型中线程的内核态和用户态?
  • 回答要点:区分内核态线程和用户态线程,并说明它们在GMP模型中的作用。
  1. Go语言中的Goroutine与线程的映射关系是怎样的?为什么选择这种映射方式?
  • 回答要点:解释Goroutine与线程的多对多映射关系及其优点。
  1. GMP模型如何解决线程调度中的锁竞争问题?
  • 回答要点:介绍全局队列和本地队列的使用,以及G的分配机制。
  1. GMP模型中的Stealing机制是什么?它如何工作?
  • 回答要点:描述Stealing机制的原理及其在Goroutine调度中的应用。
  1. 什么是Hand off机制?在什么情况下会使用该机制?
  • 回答要点:解释Hand off机制及其在阻塞和系统调用中的应用。
  1. 如何理解GMP模型中的抢占式调度?它解决了哪些问题?
  • 回答要点:说明抢占式调度的原理及其在防止协程饿死中的作用。
  1. 什么是G0和M0?它们在GMP模型中扮演什么角色?
  • 回答要点:描述G0和M0的定义及其在Goroutine调度中的功能。
  1. 请详细说明GMP模型中的调度策略。
  • 回答要点:逐步解释Goroutine的创建、唤醒、偷取、切换、自旋、系统调用和阻塞处理策略。
  1. 如何在实际项目中调优GMP调度模型?
  • 回答要点:讨论如何通过调整GOMAXPROCS等参数来优化调度性能。

带着这些问题阅读本文,可以帮助你更系统地掌握GMP模型的核心概念和调度机制,提高面试中的应答能力。

单进程时代

基本概念

在单进程时代,一个进程就是一个运行中的程序。计算机系统在执行程序时,会从头到尾依次执行完一个程序,然后再执行下一个程序。在这种模型中,不需要复杂的调度机制,因为只有一个执行流程。

面临的两个问题
  1. 单一执行流程:由于只能一个个执行程序,无法同时处理多个任务,这大大限制了CPU的利用率。
  2. 进程阻塞:当一个进程遇到I/O操作等阻塞情况时,CPU资源会被浪费,等待进程完成阻塞操作后再继续执行,导致效率低下。

多进程/线程并发时代

基本概念

为了解决单进程时代的效率问题,引入了多进程和多线程并发模型。在这种模型中,当一个进程阻塞时,CPU可以切换到另一个准备好的进程继续执行。这样可以充分利用CPU资源,提高系统的并发处理能力。

两个问题
  1. 高开销:进程拥有大量资源,进程的创建、切换和销毁都需要消耗大量的时间和资源。这导致CPU很大一部分时间都在处理进程调度,而不是实际的任务执行。
  2. 高内存占用:在32位机器下,进程的虚拟内存占用为4GB,线程占用为4MB。大量的线程和进程会导致高内存消耗,限制了系统的扩展性。
协程的引入

为了解决多进程和多线程带来的高开销和高内存占用问题,引入了协程(Coroutine)。协程是一种比线程更轻量级的执行单元。协程在用户态进行调度,避免了频繁的上下文切换带来的开销。Go语言的GMP模型正是基于协程的设计。

协程的基本概念

在深入了解Goroutine之前,先来了解一下协程(Coroutine)的基本概念。

内核态和用户态

  • 内核态线程:由操作系统管理和调度,CPU只负责处理内核态线程。
  • 用户态线程:由用户程序管理,需绑定到内核态线程上执行,协程即为用户态线程的一种。

内核态和用户态线程关系图

  • Kernel Space(内核空间):上半部分的灰色区域,表示操作系统管理的内核空间。
  • User Space(用户空间):下半部分的白色区域,表示用户程序运行的空间。
  • Kernel Thread 1 和 Kernel Thread 2(内核线程):由操作系统管理的内核线程,CPU直接处理这些线程。
  • User Thread 1、User Thread 2 和 User Thread 3(用户线程):由用户程序管理的用户线程(协程),需绑定到内核线程上执行。
执行流程
  1. 用户态线程:

    • 用户程序创建多个用户线程(如协程),如图中的“User Thread 1”、“User Thread 2”和“User Thread 3”。
  2. 内核态线程:

    • 用户线程需绑定到内核态线程上执行,如图中的“Kernel Thread 1”和“Kernel Thread 2”。
  3. CPU处理:

    • CPU只处理内核态线程,通过绑定关系,用户态线程的执行也依赖于内核态线程的调度。
    • 图中的红色箭头表示CPU正在处理内核线程,从而间接处理绑定的用户线程。

线程和协程的映射关系

  1. 单线程绑定所有协程
    • 问题1:无法利用多核CPU的能力。
    • 问题2:如果某个协程阻塞,整个线程和进程都将阻塞,导致其他协程无法执行,丧失并发能力。
  2. 一对一映射
    • 将每个协程绑定到一个线程上,退回到多进程/线程的模式,协程的创建、切换、销毁均需CPU完成,效率低下。
  3. 多对多映射
    • 允许多个协程绑定到多个线程上,形成M:N的关系。这样可以充分利用多核CPU,并通过协程调度器高效管理协程的执行。

Goroutine

Goroutine是Go语言中的协程,实现了轻量级并发。与传统的线程相比,Goroutine具有以下显著特点:

轻量级

Goroutine非常轻量,初始化时仅占用几KB的栈内存,并且栈内存可以根据需要动态伸缩。这使得我们可以在Go程序中创建成千上万个Goroutine,而不会消耗过多的系统资源。

高效调度

Goroutine的调度由Go语言的运行时(runtime)负责,而不是操作系统。Go运行时在用户态进行调度,避免了频繁的上下文切换带来的开销,使得调度更加高效。

Goroutine的使用示例

下面是一个简单的示例,展示了如何在Go语言中使用Goroutine进行并发编程。

package main

import (
    "fmt"
    "time"
)

func say(s string) {
    for i := 0; i < 5; i++ {
        time.Sleep(100 * time.Millisecond)
        fmt.Println(s)
    }
}

func main() {
    go say("Hello")
    go say("World")
    time.Sleep(1 * time.Second)
    fmt.Println("Done")
}

在这个示例中,两个Goroutine同时执行,分别打印"Hello"和"World"。通过使用go关键字,我们可以轻松地启动一个新的Goroutine。

需要注意的事项

  1. 主Goroutine的结束:在Go程序中,main函数本身也是一个Goroutine,称为主Goroutine。当主Goroutine结束时,所有其他Goroutine也会随之终止。因此,需要确保主Goroutine等待所有子Goroutine执行完毕。
  2. 同步和共享数据:虽然Goroutine之间共享内存空间,但需要通过同步机制(如通道和锁)来避免竞争条件。Go语言推荐使用通道(channel)进行Goroutine之间的通信,以保证数据的安全性和同步性。

示例:使用通道进行同步

下面的示例展示了如何使用通道来同步多个Goroutine的执行。

package main

import (
    "fmt"
    "sync"
)

func worker(id int, wg *sync.WaitGroup) {
    defer wg.Done()
    fmt.Printf("Worker %d starting\n", id)
    // 模拟工作
    fmt.Printf("Worker %d done\n", id)
}

func main() {
    var wg sync.WaitGroup

    for i := 1; i <= 5; i++ {
        wg.Add(1)
        go worker(i, &wg)
    }

    wg.Wait()
    fmt.Println("All workers done")
}

在这段代码中,使用sync.WaitGroup来同步多个Goroutine。主Goroutine启动多个子Goroutine并等待它们完成,每个子Goroutine在完成任务后调用wg.Done()减少计数,主Goroutine调用wg.Wait()阻塞等待所有子Goroutine完成。

执行流程
  1. 主Goroutine启动多个子Goroutine(Goroutine 1、2、3)。
  2. 各个Goroutine并发执行它们的任务。
  3. 每个Goroutine在完成任务后,向通道发送信号表示已完成。
  4. 主Goroutine通过通道接收所有子Goroutine的完成信号,然后继续执行。

Goroutine执行与同步流程图

这张图展示了多个Goroutine同时执行的流程以及如何通过通道(Channel)进行同步。

  • Goroutine 1、2、3:代表多个并发执行的Goroutine,分别标记为“Goroutine 1”、“Goroutine 2”和“Goroutine 3”。
  • Main Goroutine:主Goroutine,它负责启动其他Goroutine并等待它们完成。
  • Channel:用于同步Goroutine的通道。

关于waitgroup我会在下一章节中进行详细讲解,欢迎订阅我的频道!在本实例代码中大家了解使用即可。

Goroutine调度器

基本概念

在Go中,线程是运行Goroutine的实体,而调度器的功能是将可运行的Goroutine分配到工作线程上。Go语言采用了一种高效的Goroutine调度机制,使得程序能够在多核处理器上高效运行。

被废弃的调度器

早期的调度器采用了简单的设计,存在多个缺陷:

  • 概念:用大写的G表示协程,用大写的M表示线程。

  • 问题:

    • 锁竞争:每个M(线程)想要执行、放回G(协程)都必须访问一个全局G队列,因此对G的访问需要加锁以保证并发安全。当有很多线程时,锁竞争激烈,影响系统性能。
    • 局部性破坏:M转移G会造成延迟和额外的系统负载。例如,当一个G内创建另一个G’时,为了继续执行G,需要将G’交给另一个M’执行,这会破坏程序的局部性。
    • 系统开销:CPU在线程之间频繁切换导致频繁的系统调用,增加了系统开销。

GMP模型的设计思想

为了克服上述问题,Go引入了GMP模型:

  • 基本概念

  • GMP模型的组成

    • 全局G队列:存放等待运行的G。
    • P的本地G队列:存放不超过256个G。当新建协程时优先将G存放到本地队列,本地队列满了后将一半的G移动到全局队列。
    • M:内核态线程,线程想要运行协程需要先获取一个P,从P的本地G队列中获取G。当本地队列为空时,会尝试从全局队列或其他P的本地G列表中偷取G。
    • P列表:程序启动时创建GOMAXPROCS个P,并保存在数组中。
  • 调度器与OS调度器结合:Go的Goroutine调度器与操作系统调度器结合,OS调度器负责将线程分配给CPU执行。

设计策略

  • 复用线程的两个策略
    • Work Stealing机制:当本线程没有可执行的G时,优先从全局G队列中获取一批G。如果全局队列中没有,则尝试从其他P的G队列中偷取G。
    • Hand Off机制:当本线程因G进行系统调用等阻塞时,线程会释放绑定的P,把P转移给其他空闲的M执行。
  • 利用并行:有GOMAXPROCS个P,则可以有同样数量的线程并行执行。
  • 抢占式调度:Goroutine是协作式的,一个协程只有让出CPU才能让下一个协程执行,而Goroutine执行超过10ms就会强制让出CPU,防止其他协程饿死。
  • 特殊的G0和M0
    • G0:每次启动一个M都会创建的第一个Goroutine,仅用于调度,不指向任何可执行的函数。每个M都有一个自己的G0,在调度或系统调用时使用G0的栈空间。
    • M0:启动程序后的第一个主线程,负责执行初始化操作和启动第一个Goroutine,此后与其他M一样。

调度策略

  • 创建两步:

    • 通过go func()创建一个协程。
    • 新创建的协程优先保存在P的本地G队列,如果本地队列满了,会将P本地队列中的一半G打乱顺序移入全局队列。

  • 唤醒获取
    • 创建G时运行的G会尝试唤醒其他的PM组合去执行。假设G2唤醒了M2,M2绑定了P2,但P2本地队列没有G,此时M2为自旋线程。M2便会尝试从全局队列中获取G。
  • 偷取
    • 假设P的本地队列和全局队列都空了,会从其他P偷取一半G到自己的本地队列执行。
  • 切换逻辑
    • G1运行完后,M上运行的协程切换回G0,G0负责调度时协程的切换。先从P的本地队列获取G2,从G0切换到G2,从而实现M的复用。
  • 自旋
    • 自旋线程会占用CPU时间,但创建销毁线程也会消耗CPU时间,系统最多有GOMAXPROCS个自旋线程,其余的线程会在休眠M队列里。
  • 系统调用
    • 当G进行系统调用时会进入内核态被阻塞,GM会绑定在一起进行系统调用。M会释放绑定的P,把P转移给其他空闲的M执行。当系统调用结束时,GM会尝试获取一个空闲的P。
  • 阻塞处理
    • 当G因channel或network I/O阻塞时,不会阻塞M,当超过10ms时M会寻找其他可运行的G。
  • 公平性
    • 调度器每调度61次时,会尝试从全局队列里取出待运行的Goroutine来运行,如果没有找到,就去其他P偷一些Goroutine来执行。

GMP模型的优势

  1. 高效的资源利用:通过在用户态进行调度,避免了频繁的上下文切换带来的开销,充分利用CPU资源。
  2. 轻量级并发:Goroutine比线程更加轻量级,可以启动大量的Goroutine而不会消耗大量内存。
  3. 自动调度:Go运行时自动管理Goroutine的调度,无需程序员手动干预,简化了并发编程的复杂度。

面试题

如果问到了说一说GMP调度模型,建议需要说的内容

在面试中,如果被问到GMP调度模型,建议全面地回答以下内容。如果能完整且详细地讲述这些内容,将会展示你对GMP调度模型的深刻理解和熟练掌握,这将是面试中的亮点。

基本概念
  1. 线程的内核态和用户态
    • 线程分为“内核态”和“用户态”。用户态线程即协程,必须绑定一个内核态线程。CPU只负责处理内核态线程。
  2. 调度器
    • 在Go中,线程是运行Goroutine的实体。调度器的功能是将可运行的Goroutine分配到工作线程上。
  3. 映射关系
    • 在Go语言中,线程与协程的映射关系是多对多的。这样避免了多个协程对应一个线程时出现的无法使用多核和并发的问题。Go的协程是协作式的,只有让出CPU资源才能调度。如果一个协程阻塞,只有一个线程在运行,其他协程也会被阻塞。
四个概念
  1. 全局队列
    • 存放等待运行的Goroutine。
  2. 本地队列
    • 每个P(处理器)都有一个本地队列,存放不超过256个Goroutine。新建协程时优先放入本地队列,本地队列满了则将一半的G移入全局队列。
  3. GMP
    • G:Goroutine,Go语言中的协程。
    • M:Machine,内核态线程,运行Goroutine的实体。
    • P:Processor,处理器,包含运行Goroutine的资源和本地队列。
设计策略
  1. 复用线程
  • Stealing机制:当一个线程没有可执行的G时,会从全局队列或其他P的本地队列中偷取G来执行。
  • Hand off机制:当一个线程因G进行系统调用等阻塞时,线程会释放绑定的P,把P转移给其他空闲的M执行。
  1. P并行
  • 有GOMAXPROCS个P,代表最多有这么多个线程并行执行。
  1. 抢占式调度
  • Goroutine执行超过10ms就会强制让出CPU,防止其他协程饿死。
  1. 特殊的G0和M0
  • G0:每个M启动时创建的第一个Goroutine,仅用于调度,不执行用户代码。每个M都有一个G0。
  • M0:程序启动后的第一个主线程,负责初始化操作和启动第一个Goroutine。
调度策略
  1. 创建
  • 通过go func()创建一个协程。新创建的协程优先保存在P的本地G队列,如果本地队列满了,会将P本地队列中的一半G移入全局队列。
  1. 唤醒
  • 创建G时,当前运行的G会尝试唤醒其他PM组合执行。若唤醒的M绑定的P本地队列为空,M会尝试从全局队列获取G。
  1. 偷取
  • 如果P的本地队列和全局队列都为空,会从其他P偷取一半G到自己的本地队列执行。
  1. 切换
  • G1运行完后,M上运行的Goroutine切换回G0,G0负责调度协程的切换。G0从P的本地队列获取G2,实现M的复用。
  1. 自旋
  • 自旋线程会占用CPU时间,但创建销毁线程也消耗CPU时间。系统最多有GOMAXPROCS个自旋线程,其他线程在休眠M队列里。
  1. 系统调用
  • 当G进行系统调用时进入内核态被阻塞,M会释放绑定的P,把P转移给其他空闲的M执行。当系统调用结束,GM会尝试获取一个空闲的P。
  1. 阻塞处理
  • 当G因channel或network I/O阻塞时,不会阻塞M。超过10ms时,M会寻找其他可运行的G。
  1. 公平性
  • 调度器每调度61次时,会尝试从全局队列中取出待运行的Goroutine来运行。如果没有找到,就去其他P偷一些Goroutine来执行。

欢迎关注 ❤

我们搞了一个免费的面试真题共享群,互通有无,一起刷题进步。

没准能让你能刷到自己意向公司的最新面试题呢。

感兴趣的朋友们可以加我微信:wangzhongyang1993,备注:csdn面试群。

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

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

相关文章

昇思25天学习打卡营第20天|CV-ResNet50图像分类

打卡 目录 打卡 图像分类 ResNet网络介绍 数据集准备与加载 可视化部分数据集 残差网络构建 Building Block 结构 代码实现 Bottleneck结构 代码实现 构建ResNet50网络 代码定义 模型训练与评估 可视化模型预测 重点&#xff1a;通过网络层数加深&#xff0c;感知…

Windows系统使用Docker搭建Focalboard项目管理工具详细教程

目录 前言 1. 使用Docker本地部署Focalboard 1.1 在Windows中安装 Docker 1.2 使用Docker部署Focalboard 2. 安装Cpolar内网穿透工具 3. 实现公网访问Focalboard 4. 固定Focalboard公网地址 前言 作者简介&#xff1a; 懒大王敲代码&#xff0c;计算机专业应届生 今天给…

TCP为什么需要四次挥手?

tcp为什么需要四次挥手&#xff1f; 答案有两个&#xff1a; 1.将发送fin包的权限交给被动断开发的应用层去处理&#xff0c;也就是让程序员处理 2.接第一个答案&#xff0c;应用层有了发送fin的权限&#xff0c;可以在发送fin前继续向对端发送消息 为了搞清楚这个问题&…

【java】力扣 反转字符串中的单词

目录 题目描述题目描述思路代码 题目描述 151.反转字符串中的单词 题目描述 思路 主要是利用快慢指针和字符串的截取 还要了解去掉首尾空格的函数是trim 那s"the sky is blue"举例 这个例子是没有首尾空格的&#xff0c;以防万一&#xff0c;我们不管有没有&#…

第06课 Scratch入门篇:侏罗纪公园

侏罗纪公园 入门篇适合新手&#xff0c;如您已经学过&#xff0c;可以忽略本节课&#xff01; 一、故事背景&#xff1a; 在遥远的过去&#xff0c;有一个充满神秘与惊奇的时代——侏罗纪。那是一个恐龙称霸的时代&#xff0c;各种巨大的生物在这片古老的土地上留下了它们的足…

APT 安装软件详细教程

文章目录 APT 安装软件详细教程APT 概述APT 的基本命令APT 命令详解安装软件包更新和升级软件包删除软件包搜索和查找软件包管理软件包依赖清理软件包缓存APT 配置软件源配置自定义软件源常见问题及解决方案解决软件包依赖问题处理软件源错误其他常见问题使用 APT 的最佳实践总…

【设计模式】工厂模式详解

1.简介 工厂模式是一种创建型设计模式&#xff0c;通过提供一个接口或抽象类来创建对象&#xff0c;而不是直接实例化对象。工厂模式的主要思想是将对象的创建与使用分离&#xff0c;使得创建对象的过程更加灵活和可扩展。 工厂模式主要包括以下角色&#xff1a; 抽象工厂&a…

java 字符串a+b到底生成几个对象?

我们知道&#xff0c;java内存模型是堆栈元空间&#xff08;也叫方法区&#xff0c;它是在内存中的&#xff09;。 字符串常量池保存在堆里面。为了节约空间&#xff0c;如果常量池里面有&#xff0c;就不需要创建对象&#xff0c;只需要返回常量池里面的引用&#xff1b;如果…

c#实际开发长到的知识

个人建议先把rotion的库导入进来再操作,具体需要导入的库有,helper库包含了modbus通讯封装好的模块,而mvvm则可以用来做设计mvvm模块,你可以使用里面封装好的实现方法,用起来特别简单更容易实现其中的操作,但是我担心那天被卡脖子了啊啊啊,要是我罗工把库下架了那不是死…

圈子论坛小程序搭建,文章源码链接上传功能社交需求带支付功能

论坛小程序技术栈 前端&#xff1a;uni-app,vue3 后端&#xff1a;PHP,thinkphp8 数据库设计&#xff1a; 设计数据库结构&#xff0c;存储用户数据、帖子数据等。 数据库系统&#xff1a;MySQL5.7 功能开发&#xff1a; 明确小程序的功能需求&#xff1a;浏览、发帖、评…

ChemLLM:化学领域的大模型

人工智能咨询培训老师叶梓 转载标明出处 在化学这一特定学科的应用上&#xff0c;一直缺乏专门的对话模型。化学数据和科学知识通常存储在结构化的数据库中&#xff0c;这给直接使用这些数据训练语言模型带来了挑战。为了解决这一问题&#xff0c;来自上海人工智能实验室的研究…

RabbitMQ高级篇(如何保证消息的可靠性、如何确保业务的幂等性、延迟消息的概念、延迟消息的应用)

文章目录 1. 消息丢失的情况2. 生产者的可靠性2.1 生产者重连2.2 生产者确认2.3 生产者确认机制的代码实现2.4 如何看待和处理生产者的确认信息 3. 消息代理&#xff08;RabbitMQ&#xff09;的可靠性3.1 数据持久化3.2 LazyQueue&#xff08; 3.12 版本后所有队列都是 Lazy Qu…

《七日世界》游玩感想

《七日世界》是一款最近新出的引人入胜的游戏&#xff0c;它以独特的故事情节和精美的画面设计吸引了许多玩家的关注。在这款游戏中&#xff0c;玩家需要在七天的时间里探索一个神秘的幽灵世界&#xff0c;解开其中的谜题&#xff0c;救出被困的灵魂。 首先&#xff0c;让我来聊…

1996-2023年上市公司绿色并购数据(含原始数据+处理代码+计算结果)

1996-2023年上市公司绿色并购数据&#xff08;含原始数据处理代码计算结果&#xff09; 1、时间&#xff1a;1996-2023年 2、指标&#xff1a;股票代码、首次公告日期、年份、买方、卖方、标的方、交易概述、标的物名称、标的物说明、买方经营范围、卖方经营范围、标的方经营…

28.jdk源码阅读之CopyOnWriteArraySet

1. 写在前面 CopyOnWriteArraySet 是 Java 中一个线程安全的 Set 实现&#xff0c;它的底层是基于 CopyOnWriteArrayList 实现的。这种数据结构在并发编程中非常有用&#xff0c;因为它在写操作时会创建一个新的数组副本&#xff0c;从而避免了并发修改问题。不知道大家对它的底…

angular入门基础教程(五)父子组件的数据通信

组件之间的通信是我们业务开发中少不了的,先了解下父子组件的通信 父组件传数据给子组件 前面&#xff0c;我们学会会动态属性的绑定&#xff0c;所以在父组件中给子组件绑定属性&#xff0c;在子组件中就可以使用这个属性了。 父组件中声明然后赋值 export class AppCompon…

C语言 | Leetcode C语言题解之第304题二维区域和检索-矩阵不可变

题目&#xff1a; 题解&#xff1a; typedef struct {int** sums;int sumsSize; } NumMatrix;NumMatrix* numMatrixCreate(int** matrix, int matrixSize, int* matrixColSize) {NumMatrix* ret malloc(sizeof(NumMatrix));ret->sums malloc(sizeof(int*) * (matrixSize …

图论:721. 账户合并(并查集扩展)

文章目录 1、题目链接2、题目描述3、并查集思路3.1、按秩合并3.2、常用并查集代码 4、题目解析 1、题目链接 721. 账户合并 2、题目描述 3、并查集思路 并查集可以在很短的时间内合并不同的集合。它的思想为&#xff0c;一开始将不同单元单独作为一个结点&#xff0c;然后按…

【Qt】修改窗口的标题和图标

以下操作仅对顶层 widget(独⽴窗口),有效。 修改窗口的标题 一.windowTitle属性 1.概念 是一种在用户界面中显示窗口的标题的属性。它可以用来设置窗口的标题栏文本。 2.API API说明windowTitle()获取到控件的窗⼝标题.setWindowTitle(const QString& title)设置控件的…