【图论】图的C++实现代码

在这个例程中我们用类实现了节点、(无向图)连边、无向图,实现了节点度的计算、无向图聚类系数计算、度分布统计、无向图的Dijkstra算法(已知起止点计算最短路的算法)

#include <iostream>
#include<vector>
#include<set>
#include<unordered_map>
using namespace std;class Edge
{
public:int v1;int v2;int weight;Edge(int v1, int v2, int w){this->v1 = v1;this->v2 = v2;this->weight = w;}
};class DJPathInfo // 存放已知最短路
{
public:vector<vector<int>> shortest_path;vector<int> shortest_path_len;DJPathInfo(int n){vector<int> empty;this->shortest_path=vector<vector<int>>(n, empty);this->shortest_path_len= vector<int>(n, 0);}
};class Node
{
public:std::vector<Edge> edges;int id;Node(int i){this->id = i;}int degree(){return this->edges.size();}vector<int> neighbors(){vector<int> res;if (this->edges.size() > 0){for (auto edge : this->edges){if (edge.v1 == this->id) res.push_back(edge.v2);else res.push_back(edge.v1);}return res;}else return res;}float cluster_coeff(vector<vector<bool>> adj_mat){int num_neighbor = this->edges.size() - 1;if (num_neighbor < 2) return 0;vector<int> neighbors = this->neighbors();float E = 0; //邻居之间的边数for (auto i : neighbors){for (auto j : neighbors){if (i >= j) continue;if (adj_mat[i][j] == true) E++;}}cout << 2 * E / (neighbors.size() * (neighbors.size() - 1)) << endl;return 2 * E / (neighbors.size() * (neighbors.size() - 1));}
};class Graph
{
public:vector<vector<bool>> adj_mat;vector<vector<float>> w_mat;int node_num;vector<Edge> edges;vector<Node> nodes;Graph(vector<vector<bool>> A, vector<vector<float>> W){this->adj_mat = A;this->w_mat = W;this->node_num = A.size();for (int i = 0; i < node_num; i++) this->nodes.push_back(Node(i));for (int i = 0; i < node_num; i++)for (int j = 0; j < i; j++){if (A[i][j] == true) this->add_edge(i, j, W[i][j]);}}Graph(int n){vector<float> zero(n, 0);vector<bool> all_false(n, false);vector<vector<bool>> A(n, all_false);vector<vector<float>> W(n, zero);this->adj_mat = A;this->w_mat = W;this->node_num = A.size();for (int i = 0; i < node_num; i++) this->nodes.push_back(Node(i));for (int i = 0; i < node_num; i++)for (int j = 0; j < i; j++){if (A[i][j] == true) this->add_edge(i, j, W[i][j]);}}void add_edge(int id1, int id2, int weight){Edge e = Edge(id1, id2, weight);this->adj_mat[id1][id2] = true;this->adj_mat[id2][id1] = true;this->w_mat[id1][id2] = weight;this->w_mat[id2][id1] = weight;this->edges.push_back(e);this->nodes[id1].edges.push_back(e);this->nodes[id2].edges.push_back(e);}void tell_info(){for (auto v : this->nodes){std::cout << "Node ID:" << v.id << std::endl;std::cout << "Neighbor/Weight:";for (auto i : v.neighbors()) cout <<'(' << i<<','<< this->w_mat[v.id][i] << ')' << ',';std::cout<<std::endl;}}vector<int> DJ(int s, int dest) // Dijkstra算法{DJPathInfo info(this->nodes.size());vector<int> res = {s};set<int> S;S.insert(s);set<int> U;for (auto v : this->nodes[s].neighbors()) U.insert(v);if (U.size() == 0) return res;for (auto v : U){info.shortest_path[v].push_back(v);info.shortest_path_len[v] = this->w_mat[s][v];}info.shortest_path_len[s] = 0;while (U.size() != 0){int n = *U.begin();int n_len = info.shortest_path_len[n];for (auto it = U.begin(); it != U.end(); it++){if (info.shortest_path_len[n] > info.shortest_path_len[*it]){n = *it;n_len = info.shortest_path_len[*it];}}S.insert(n);U.erase(n);for (auto v : this->nodes[n].neighbors()){if (S.find(v) != S.end()) continue;U.insert(v);if (info.shortest_path[v].size() == 0){info.shortest_path[v] = info.shortest_path[n];info.shortest_path[v].push_back(v);info.shortest_path_len[v] = info.shortest_path_len[n] + this->w_mat[n][v];continue;}if (info.shortest_path_len[n] + this->w_mat[n][v] < info.shortest_path_len[v]){info.shortest_path[v] = info.shortest_path[n];info.shortest_path[v].push_back(v);info.shortest_path_len[v] = info.shortest_path_len[n] + this->w_mat[n][v];}}}if (S.find(dest) != S.end()){for (auto v : info.shortest_path[dest]) res.push_back(v);info.shortest_path_len[dest];}return res;}float cluster_coeff(){float res=0;for (auto node : this->nodes) res += node.cluster_coeff(this->adj_mat);return res / float(this->nodes.size());}void degree_distribution() // 度分布{vector<float> res;unordered_map<int, int> degree_count;for (auto v : this->nodes){degree_count.emplace(v.degree(), 0);}for (auto v : this->nodes){degree_count[v.degree()]++;}for (auto it : degree_count){std::cout << "degree =" << it.first << ', '<<" prob=" << float(it.second) / this->nodes.size() << endl;}}
};int main()
{Graph G = Graph(11);G.add_edge(0, 1, 2);G.add_edge(0, 2, 9);G.add_edge(0, 3, 1);G.add_edge(1, 4, 1);G.add_edge(1, 2, 6);G.add_edge(2, 4, 5);G.add_edge(2, 5, 1);G.add_edge(2, 6, 2);G.add_edge(2, 3, 7);G.add_edge(3, 6, 9);G.add_edge(4, 7, 2);G.add_edge(4, 8, 9);G.add_edge(4, 5, 3);G.add_edge(5, 8, 6);G.add_edge(5, 6, 4);G.add_edge(6, 8, 3);G.add_edge(6, 9, 1);G.add_edge(7, 8, 7);G.add_edge(7, 10, 9);G.add_edge(8, 10, 2);G.add_edge(8, 9, 1);G.add_edge(9, 10, 4);G.tell_info();cout << "Djkstra Test: shortest 0 -> 10: ";for (const auto& p : G.DJ(0, 10)) cout << p << ',';cout<<endl;cout << "clustring coefficient=" << G.cluster_coeff() << endl;cout << "degree distribution:"  << endl;G.degree_distribution();return 0;
}

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

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

相关文章

Java:二维数组

目录 1. 二维数组的基础格式 1.1 二维数组变量的创建 —— 3种形式 1.2 二维数组的初始化 \1 动态初始化 \2 静态初始化 2. 二维数组的大小 和 内存分配 3. 二维数组的不规则初始化 4. 遍历二维数组 4.1 for循环 ​编辑 4.2 for-each循环 5. 二维数组 与 方法 5.1…

Code::Blocks 24.10 全中文优化完整版

Code::Blocks&#xff08;或者叫做 CodeBlocks&#xff09;是一款开放源代码、跨平台的集成开发环境&#xff08;IDE&#xff09;&#xff0c;通过配置不同的编程语言编译器&#xff0c;可以用于多种编程语言程序开发。 网上有很多文章介绍 Code::Blocks 的安装&#xff0c;通…

分组校验在Spring中的应用详解

目录 前言1. 什么是分组校验2. 分组校验的基本原理3. 分组校验的实现步骤3.1 定义分组接口3.2 在校验项中指定分组3.3 校验时指定要校验的分组3.4 默认分组和分组的继承 4. 分组校验的优势和适用场景4.1 优势4.2 适用场景 5. 常见问题与解决方案5.1 校验未生效5.2 无法识别默认…

优化前端开发中的提示语设计基本原则

文章目录 一致1、同一对象&#xff0c;指称一致2、同一状态&#xff0c;描述一致3、同一行为&#xff0c;提示一致 简洁1、用词简短 条理1、上下呼应2、主次分明 亲和1、化“难”为易2、“礼”字当先3、正向表达 灵动1、用词多变2、远离平淡 契合1、身份契合2、产品契合 示例1、…

C++知识点总结(56):数学专题

数学专题 一、进制转换类1. 模板1.1 十转 x x x1.2 x x x 转十 2. 例题 二、公式推导类1. 一元二次方程解2. 例题2.1 【模板】怪物同笼2.2 K K K 的倍数 三、枚举例题1. 二次方程2. 【模板】立方体体积3. 街头篮球 一、进制转换类 1. 模板 1.1 十转 x x x while(n){num[…

【C++滑动窗口】1297. 子串的最大出现次数|1748

本文涉及的基础知识点 C算法&#xff1a;滑动窗口及双指针总结 固定长度滑动窗口 LeetCode1297. 子串的最大出现次数 给你一个字符串 s &#xff0c;请你返回满足以下条件且出现次数最大的 任意 子串的出现次数&#xff1a; 子串中不同字母的数目必须小于等于 maxLetters 。…

【C++练习】使用海伦公式计算三角形面积

编写并调试一个计算三角形面积的程序 要求&#xff1a; 使用海伦公式&#xff08;Herons Formula&#xff09;来计算三角形的面积。程序需要从用户那里输入三角形的三边长&#xff08;实数类型&#xff09;。输出计算得到的三角形面积&#xff0c;结果保留默认精度。提示用户…

Python使用PDF相关组件案例详解

主要对pdfminer.six、pdfplumber、PyMuPDF、PyPDF2、PyPDF4、pdf2image、camelot-py七个PDF相关组件分别详解&#xff0c;具体使用案例演示 1. pdfminer.six pdfminer.six 是一个专门用来从 PDF 中提取文本的库&#xff0c;能够处理复杂的文本布局&#xff0c;适合用于文本解析…

计算机网络:网络层 —— 移动 IP 技术

文章目录 移动性对因特网应用的影响移动 IP 相关基本概念移动IP技术的基本工作原理代理发现与注册固定主机向移动主机发送IP数据报移动主机向固定主机发送IP数据报同址转交地址方式三角形路由问题 移动性对因特网应用的影响 我们列举如下三个应用场景说明移动性对因特网应用的…

鸿蒙多线程开发——Worker多线程

1、概 述 1.1、基本介绍 Worker主要作用是为应用程序提供一个多线程的运行环境&#xff0c;可满足应用程序在执行过程中与主线程分离&#xff0c;在后台线程中运行一个脚本进行耗时操作&#xff0c;极大避免类似于计算密集型或高延迟的任务阻塞主线程的运行。 创建Worker的线…

【大数据学习 | kafka】消费者的分区分配规则

1. 概述 上面我们提到过&#xff0c;消费者有的时候会少于或者多于分区的个数&#xff0c;那么如果消费者少了有的消费者要消费多个分区的数据&#xff0c;如果消费者多了&#xff0c;有的消费者就可能没有分区的数据消费。 那么这个关系是如何分配的呢&#xff1f; 现在我们…

Python接口自动化测试自学指南(项目实战)

&#x1f345; 点击文末小卡片 &#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 接口自动化测试是指通过编写程序来模拟用户的行为&#xff0c;对接口进行自动化测试。Python是一种流行的编程语言&#xff0c;它在接口自动化测试中得到了广…

【Python进阶】Python网络协议与套接字编程:构建客户端和服务器

1、网络通信基础与网络协议 1.1 网络通信模型概述 网络通信是信息时代基石&#xff0c;它如同现实世界中的邮递系统&#xff0c;将数据从一处传递到另一处。其中&#xff0c;OSI七层模型与TCP/IP四层或五层模型是理解和构建网络通信的基础。 1.1.1 OSI七层模型与TCP/IP四层/…

Redis - 哨兵(Sentinel)

Redis 的主从复制模式下&#xff0c;⼀旦主节点由于故障不能提供服务&#xff0c;需要⼈⼯进⾏主从切换&#xff0c;同时⼤量 的客⼾端需要被通知切换到新的主节点上&#xff0c;对于上了⼀定规模的应⽤来说&#xff0c;这种⽅案是⽆法接受的&#xff0c; 于是Redis从2.8开始提…

UE5 metahuman 头发物理模拟

https://www.youtube.com/watch?vyYmdgtP6cQA 头发梳理物理 打开Unreal Engine&#xff0c;选择一个角色模型 点击“Groom”选项卡&#xff0c;在“Physics”部分下&#xff0c;找到“Custom Solver”&#xff0c;点击下拉菜单&#xff0c;选择“WindDrivenSystem”。 在“…

大数据 ETL + Flume 数据清洗 — 详细教程及实例(附常见问题及解决方案)

大数据 ETL Flume 数据清洗 — 详细教程及实例 1. ETL 和 Flume 概述1.1 ETL&#xff08;Extract, Transform, Load&#xff09;1.2 Flume 概述 2. Flume 环境搭建2.1 下载并安装 Flume2.2 启动 Flume 3. Flume 配置和常见 Source、Sink、Channel3.1 Flume Source3.2 Flume Si…

24年配置CUDA12.4,Pytorch2.5.1,CUDAnn9.5运行环境

没什么好介绍的&#xff0c;直接说了。 下载 首先打开命令行&#xff0c;输入代码查看显卡最高支持的cuda版本&#xff0c;下载的版本不要高于该版本 nvidia-smi PyTorch 插件这个是PyTorch下载地址&#xff0c;就按照我这么选CUDA版本就选最新的&#xff0c;看好绿框里的CU…

架构师备考-概念背诵(系统架构)

软件架构概念 一个程序和计算系统软件体系结构是指系统的一个或者多个结构。结构中包括软件的构件,构件的外部可见属性以及它们之间的相互关系。体系结构并非可运行软件。确切地说,它是一种表达,使软件工程师能够: (1)分析设计在满足所规定的需求方面的有效性:(2)在设计变…

Linux服务器软件包管理的使用

在 Linux 系统中&#xff0c;软件包管理器是用于安装、升级、删除和管理软件包的工具。不同的 Linux 发行版使用不同的软件包管理器&#xff0c;通常根据使用的包格式和包管理系统&#xff08;如 .deb 或 .rpm&#xff09;来区分。下面将介绍几种常见的 Linux 软件包管理器及其…

debian系统安装qt的时候 显示xcb相关文件缺失

如果是安装之后的问题 我们可以选择使用ldd的命令查看当前依赖的so那些文件确实 ldd /home/yinsir/Qt/5.15.2/gcc_64/plugins/platforms/libqxcb.so 本人在进行打包的时候 出现则会个报错 ERROR: ldd outputLine: “libxcb-util.so.1 > not found” ERROR: for binary: “/…