算法基础学习笔记——⑫最小生成树\二分图\质数\约数

博主:命运之光
专栏:算法基础学习

在这里插入图片描述

目录

 ✨最小生成树 

🍓朴素Prim

🍓Kruskal算法

✨二分图

🍓匈牙利算法

 ✨质数

🍓(1)质数的判定——试除法

🍓(2)分解质因数——试除法

✨约数

🍓(1)试除法求一个数的所有约数

🍓(2)约数个数

🍓(3)约数之和

🍓(4)欧几里得算法(辗转相除法)


前言:算法学习笔记记录日常分享,需要的看哈O(∩_∩)O,感谢大家的支持!


 ✨最小生成树 

🍓朴素Prim

🍓朴素版prim算法:

时间复杂度是 O(n2+m)O(n2+m), nn 表示点数,mm 表示边数

int n; // n表示点数
int g[N][N]; // 邻接矩阵,存储所有边
int dist[N]; // 存储其他点到当前最小生成树的距离
bool st[N]; // 存储每个点是否已经在生成树中
// 如果图不连通,则返回INF(值是0x3f3f3f3f), 否则返回最小生成树的树边权重之和
int prim()
{
     memset(dist, 0x3f, sizeof dist);
     int res = 0;
     for (int i = 0; i < n; i ++ )
     {
         int t = -1;
         for (int j = 1; j <= n; j ++ )
             if (!st[j] && (t == -1 || dist[t] > dist[j]))
             	t = j;
         if (i && dist[t] == INF) return INF;
         if (i) res += dist[t];
         st[t] = true;
         for (int j = 1; j <= n; j ++ ) dist[j] = min(dist[j], g[t][j]);
     }
     return res;
}

🍓Kruskal算法

Kruskal算法:

时间复杂度是 O(mlogm)O(mlogm), nn 表示点数,mm 表示边数

int n, m; // n是点数,m是边数
int p[N]; // 并查集的父节点数组
struct Edge // 存储边
{
     int a, b, w;
     bool operator< (const Edge &W)const
     {
     	return w < W.w;
     }
}edges[M];
int find(int x) // 并查集核心操作
{
     if (p[x] != x) p[x] = find(p[x]);
     return p[x];
}
int kruskal()
{
     sort(edges, edges + m);
     for (int i = 1; i <= n; i ++ ) p[i] = i; // 初始化并查集
     int res = 0, cnt = 0;
     for (int i = 0; i < m; i ++ )
     {
         int a = edges[i].a, b = edges[i].b, w = edges[i].w;
         a = find(a), b = find(b);
         if (a != b) // 如果两个连通块不连通,则将这两个连通块合并
         {
             p[a] = b;
             res += w;
             cnt ++ ;
         }
     }
     if (cnt < n - 1) return INF;
     return res;
}

✨二分图

染色法

判断一个图是不是二分图

二分图:可以把所有点分成两边,使所有边在集合之间,集合内部没有边。

二分图当且仅当图中不含奇数环

🍓染色法判别二分图:

时间复杂度是 O(n+m)O(n+m), nn 表示点数,mm 表示边数

int n; // n表示点数
int h[N], e[M], ne[M], idx; // 邻接表存储图
int color[N]; // 表示每个点的颜色,-1表示未染色,0表示白色,1表示黑色
// 参数:u表示当前节点,c表示当前点的颜色
bool dfs(int u, int c)
{
 	 color[u] = c;
     for (int i = h[u]; i != -1; i = ne[i])
     {
         int j = e[i];
         if (color[j] == -1)
         {
         	if (!dfs(j, !c)) return false;
         }
         else if (color[j] == c) return false;
     }
     return true;
}
bool check()
{
     memset(color, -1, sizeof color);
     bool flag = true;
     for (int i = 1; i <= n; i ++ )
         if (color[i] == -1)
             if (!dfs(i, 0))
             {
                 flag = false;
                 break;
             }
     return flag;
}

🍓匈牙利算法

🍓匈牙利算法:

时间复杂度是 O(nm)O(nm), nn 表示点数,mm 表示边数

int n1, n2; // n1表示第一个集合中的点数,n2表示第二个集合中的点数
int h[N], e[M], ne[M], idx; // 邻接表存储所有边,匈牙利算法中只会用到从第一个集合指向第二个集合的边,所以这里只用存一个方向的边
int match[N]; // 存储第二个集合中的每个点当前匹配的第一个集合中的点是哪个
bool st[N]; // 表示第二个集合中的每个点是否已经被遍历过
bool find(int x)
{
     for (int i = h[x]; i != -1; i = ne[i])
     {
         int j = e[i];
         if (!st[j])
         {
             st[j] = true;
             if (match[j] == 0 || find(match[j]))
             {
                 match[j] = x;
                 return true;
             }
         }
     }
     return false;
}
// 求最大匹配数,依次枚举第一个集合中的每个点能否匹配第二个集合中的点
int res = 0;
for (int i = 1; i <= n1; i ++ )
{
     memset(st, false, sizeof st);
     if (find(i)) res ++ ;
}

 ✨质数

🍓所有大于1的自然数,所有<=1的数既不是质数也不是合数

定义:在大于1的整数中,如果只包含1和本身这两个约数,就被称为质数,或者叫素数

🍓(1)质数的判定——试除法

质数的一个重要性质:如果d能整除n,显然n除d也能整除n

故发现n的所有的约数都是成对出现的(d与n/d都成成对出现的)

所以枚举时可以只枚举每一对当中较小的那一个,枚举:

🍓试除法判定质数:

bool is_prime(int x)
{
     if (x < 2) return false;
     for (int i = 2; i <= x / i; i ++ )
     	if (x % i == 0)
     		return false;
     return true;
}

🍓(2)分解质因数——试除法

从小到大枚举所有数

🍓试除法分解质因数:

void divide(int x)
{
     for (int i = 2; i <= x / i; i ++ )
         if (x % i == 0)
         {
             int s = 0;
             while (x % i == 0) x /= i, s ++ ;
             cout << i << ' ' << s << endl;
         }
     if (x > 1) cout << x << ' ' << 1 << endl;
     cout << endl;
}

🍓

罗列出每个数,依次删除每个数的倍数,剩下的数就是质数,可以对此进行优化,可以不删每一个数的倍数, 可以只删质数的倍数,这样就不用重复删。

🍓质数定理:

优化完的筛法:埃氏筛法

🍓朴素筛法求素数:

int primes[N], cnt; // primes[]存储所有素数
bool st[N]; // st[x]存储x是否被筛掉
void get_primes(int n)
{
     for (int i = 2; i <= n; i ++ )
     {
         if (st[i]) continue;
         primes[cnt ++ ] = i;
         for (int j = i + i; j <= n; j += i)
         	st[j] = true;
     }
}

🍓线性筛法:

把每一个合数用它的某个质因子筛掉

每个数都会被其最小质因子筛掉,而且每个数只有一个最小质因子,故每个数只会被筛一次

🍓线性筛法求素数:

int primes[N], cnt; // primes[]存储所有素数
bool st[N]; // st[x]存储x是否被筛掉
void get_primes(int n)
{
     for (int i = 2; i <= n; i ++ )
     {
         if (!st[i]) primes[cnt ++ ] = i;
         for (int j = 0; primes[j] <= n / i; j ++ )
         {
             st[primes[j] * i] = true;
             if (i % primes[j] == 0) break;
         }
     }
}

约数

约数

🍓(1)试除法求一个数的所有约数

🍓试除法求所有约数:

vector<int> get_divisors(int x)
{
     vector<int> res;
     for (int i = 1; i <= x / i; i ++ )
         if (x % i == 0)
         {
             res.push_back(i);
             if (i != x / i) res.push_back(x / i);
         }
     sort(res.begin(), res.end());
     return res;
}

🍓(2)约数个数

🍓(3)约数之和

约数个数和约数之和:

如果 N = p1^c1 * p2^c2 * ... *pk^ck

约数个数: (c1 + 1) * (c2 + 1) * ... * (ck + 1)

约数之和: (p1^0 + p1^1 + ... + p1^c1) * ... * (pk^0 + pk^1 + ... + pk^ck)

🍓(4)欧几里得算法(辗转相除法)

🍓欧几里得算法:

int gcd(int a, int b)
{
	return b ? gcd(b, a % b) : a; // <表达式1>?<表达式2>:<表达式3>,
} //它的意思是,如果表达式1成立,则输出表达式2的值,否则输出表达式3的值

补充小知识:

两个数的积等于它们最大公约数和它们最小公倍数的

积。公式表示为 :a×b=gcd(a,b)×lcm(a,b)

🍓最小公倍数与最大公约数模板:

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

(转载)基于遗传算法的多目标优化算法(matlab实现)

1 理论基础 1.1 多目标优化及Pareto最优解 多目标优化问题可以描述如下&#xff1a; 其中&#xff0c;f(x)为待优化的目标函数&#xff1b;x为待优化的变量&#xff1b;Ib和ub分别为变量x的下限和上限约束&#xff1b;Aeq*xbeq为变量x的线性等式约束&#xff1b;A*x≤b为变…

数据库作业

目录 数据库teaching中的表结构和表记录。 问题&#xff1a; 答案&#xff1a; 数据库teaching中的表结构和表记录。 &#xff08;1&#xff09;学生信息表student    #student表结构      create table if not exists student (      studentno char(11) not…

c++ 11标准模板(STL) std::map(六)

定义于头文件<map> template< class Key, class T, class Compare std::less<Key>, class Allocator std::allocator<std::pair<const Key, T> > > class map;(1)namespace pmr { template <class Key, class T, clas…

【Linux驱动】认识驱动(驱动的概念、驱动分类)

目录 1、什么是驱动&#xff1f; 2、应用程序调用驱动基本流程 3、file_operations 结构体 4、驱动的分类 1、什么是驱动&#xff1f; 驱动就是一段程序&#xff0c;能够获取外设或者传感器数据、控制外设。驱动获取到的数据会提交给应用程序。 在 Linux 中一切皆为文件&…

物联网GPRS模块流量计算

物联网GPRS模块流量计算 MQTT(消息队列遥测传输) 是ISO 标准下一个基于TCP/IP的消息发布/订阅传输协议。 一、TCP消耗流量计算 以太网数据包结构&#xff1a; 以太网首部 IP首部 TCP首部 APPL首部 用户数据 以太网尾部 以太网首部为14个字节 IP首部为20个字节 TCP首部…

【CesiumJS入门】(1)创建Viewer及相关配置项

前言 在上一篇博客中&#xff0c;我们直接在vue组件完成初始渲染并创建 DOM 节点后通过 const map new Cesium.Viewer(cesiumContainer)构建了一个地球场景。 而本篇&#xff0c;我们将会专门把地球创建的方法写在一个js文件中&#xff0c;以便后续的调用。 同时&#xff0…

【JavaSE】Java基础语法(三十二):Stream流

文章目录 1. 体验Stream流2. Stream流的常见生成方式3. Stream流中间操作方法【应用】4. Stream流终结操作方法【应用】5. Stream流的收集操作 1. 体验Stream流 案例需求 按照下面的要求完成集合的创建和遍历 创建一个集合&#xff0c;存储多个字符串元素把集合中所有以"…

torch_scatter.scatter()的使用方法

学习目标&#xff1a; 在学习PyG时&#xff0c;遇到了 scatter 这个函数&#xff0c;经过学习加上自身的理解&#xff0c;记录如下以备复习 学习内容&#xff1a; src&#xff1a;表示输入的tensor&#xff0c;接下来被处理&#xff1b;index&#xff1a;表示tensor对应的索引…

机器学习 day14 ( 神经网络 )

神经网络的发展 最开始的动机&#xff1a;是通过构建软件来模拟大脑&#xff0c;但今天的神经网络几乎与大脑的学习方式无关 我们依据大脑中的神经网络&#xff0c;来构建人工神经网络模型。左图中&#xff1a;一个神经元可以看作一个处理单元&#xff0c;它有很多的输入/树突…

chatgpt赋能python:Python创建一个Animal类介绍

Python创建一个Animal类介绍 Python是一种高级编程语言&#xff0c;其简单易学、灵活性强、可读性高以及强大的库使得Python非常受欢迎。在Python中创建类非常容易且非常常见&#xff0c;我们可以使用Python创建各种类型的类。今天&#xff0c;我们将讨论如何使用Python创建一…

【时空权衡】

目录 知识框架No.0 时空权衡一、基本思想 No.1 计数排序一、比较计数二、分布计数 No.2 散列法一、开散列&#xff08;分离链&#xff09;二、闭散列&#xff08;开式寻址&#xff09; 知识框架 No.0 时空权衡 一、基本思想 其实时空权衡&#xff1a;是指在算法的设计中&…

进程信号(Linux)

进程信号 信号入门身边的信号进程信号 产生信号终端按键产生信号调用系统函数向目标进程发信号killraiseabort 硬件异常产生信号由软件条件产生信号 阻塞信号信号其他相关常见概念在内核中的表示sigset_t信号集操作函数sigprocmasksigpending 捕捉信号内核如何实现信号的捕捉si…

分布式简要说明

1.分布式简要说明 《分布式系统原理与范型》定义&#xff1a; 分布式系统是若干独立计算机的集合&#xff0c;这些计算机对于用户来说就像单个相关系统。 分布式系统 (distributed system) 是建立在网络之上的软件系统。 随着互联网的发展&#xff0c;网站应用的规模不断扩…

SAP-QM-物料主数据-质量管理视图字段解析

过账到质检库存&#xff1a;要勾选&#xff0c;否则收货后库存不进入质检库存HU检验&#xff1a;收货到启用HU管理的库位时产生检验批&#xff0c;例如某个成品物料是收货到C002库位&#xff0c;该库位启用了HU管理&#xff0c;那么此处要勾选。但是如果勾选了&#xff0c;却收…

04.hadoop上课笔记之java编程和hbase

1.win查看服务 netstat -an #linux也有#R数学建模语言 SCALAR 2.java连接注意事项,代码要设置用户 System.setProperty("HADOOP_USER_NAME", "hadoop");3.伪分布式的好处(不用管分布式细节,直接连接一台机器…,适合用于学习) 4.官方文档 查看类(static |…

Python期末复习题库(下)——“Python”

小雅兰期末加油冲冲冲&#xff01;&#xff01;&#xff01; 1. (单选题)下列关于文件打开模式的说法,错误的是( C )。 A. r代表以只读方式打开文件 B. w代表以只写方式打开文件 C. a代表以二进制形式打开文件 D. 模式中使用时,文件可读可写 2. (单选题)下列选项中,以追加…

webpack的使用

一、什么是webpack&#xff1f; webpack是一个前端构建工具&#xff0c;目前比较主流的构建工具&#xff0c;自定义的模块比较多。 二、应用场景 vue、react、angular 都可以通过webpack构建全部可供访问的页面数量不超过500个 三、安装 通过npm方式在项目根目录下执行命令…

htmlCSS-----CSS选择器(下)

目录 前言&#xff1a; 2.高级选择器 &#xff08;1&#xff09;子代选择器 &#xff08;2&#xff09;伪类选择器 &#xff08;3&#xff09;后代选择器 &#xff08;4&#xff09;兄弟选择器 相邻兄弟选择器 通用兄弟选择器 &#xff08;5&#xff09;并集选择器 &am…

【JavaSE】Java基础语法(二十六):Collection集合

文章目录 1. 数组和集合的区别2. 集合类体系结构3. Collection 集合概述和使用【应用】4. Collection集合的遍历【应用】5. 增强for循环【应用】 1. 数组和集合的区别 相同点 都是容器,可以存储多个数据不同点 数组的长度是不可变的,集合的长度是可变的 数组可以存基本数据类型…

基于Yarn搭建Flink

基于Yarn搭建Flink 1. 概述 1.1 Yarn 简介 Apache Hadoop YARN是一个资源提供程序&#xff0c;受到许多数据处理框架的欢迎。Flink服务被提交给 YARN 的 ResourceManager&#xff0c;后者再由 YARN NodeManager 管理的机器上生成容器。Flink 将其 JobManager 和 TaskManager…