news 2026/5/11 23:42:19

MPI 广播一个数组代码示例

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
MPI 广播一个数组代码示例

1.基本广播操作 (MPI_Bcast)

#include <stdio.h> #include <stdlib.h> #include <mpi.h> int main(int argc, char** argv) { MPI_Init(&argc, &argv); int rank, size; MPI_Comm_rank(MPI_COMM_WORLD, &rank); MPI_Comm_size(MPI_COMM_WORLD, &size); int n = 10; // 数组长度 int* array = NULL; // 根进程(通常是 rank 0)初始化数组 if (rank == 0) { array = (int*)malloc(n * sizeof(int)); for (int i = 0; i < n; i++) { array[i] = i * 10; // 示例数据 } printf("Rank %d: Initialized array: ", rank); for (int i = 0; i < n; i++) { printf("%d ", array[i]); } printf("\n"); } else { // 其他进程分配内存 array = (int*)malloc(n * sizeof(int)); } // 广播数组长度(先广播长度,再广播数据) MPI_Bcast(&n, 1, MPI_INT, 0, MPI_COMM_WORLD); // 广播整个数组 MPI_Bcast(array, n, MPI_INT, 0, MPI_COMM_WORLD); // 所有进程打印接收到的数组 printf("Rank %d: Received array: ", rank); for (int i = 0; i < n; i++) { printf("%d ", array[i]); } printf("\n"); free(array); MPI_Finalize(); return 0; }

2.动态数组广播(长度未知)

#include <stdio.h> #include <stdlib.h> #include <mpi.h> int main(int argc, char** argv) { MPI_Init(&argc, &argv); int rank, size; MPI_Comm_rank(MPI_COMM_WORLD, &rank); MPI_Comm_size(MPI_COMM_WORLD, &size); int* array = NULL; int n = 0; // 根进程确定数组长度和内容 if (rank == 0) { // 假设根据某些条件确定长度 n = size * 3; // 示例:每个进程处理3个元素 array = (int*)malloc(n * sizeof(int)); for (int i = 0; i < n; i++) { array[i] = i * 5; } printf("Root: Broadcasting array of size %d\n", n); } // 第一步:广播数组长度 MPI_Bcast(&n, 1, MPI_INT, 0, MPI_COMM_WORLD); // 非根进程分配内存 if (rank != 0) { array = (int*)malloc(n * sizeof(int)); } // 第二步:广播数组数据 MPI_Bcast(array, n, MPI_INT, 0, MPI_COMM_WORLD); // 验证广播结果 printf("Rank %d: First element = %d, Last element = %d\n", rank, array[0], array[n-1]); free(array); MPI_Finalize(); return 0; }

3.二维数组广播

#include <stdio.h> #include <stdlib.h> #include <mpi.h> int main(int argc, char** argv) { MPI_Init(&argc, &argv); int rank; MPI_Comm_rank(MPI_COMM_WORLD, &rank); int rows = 3, cols = 4; int** matrix = NULL; // 根进程初始化矩阵 if (rank == 0) { matrix = (int**)malloc(rows * sizeof(int*)); for (int i = 0; i < rows; i++) { matrix[i] = (int*)malloc(cols * sizeof(int)); for (int j = 0; j < cols; j++) { matrix[i][j] = i * cols + j; } } printf("Root matrix:\n"); for (int i = 0; i < rows; i++) { for (int j = 0; j < cols; j++) { printf("%2d ", matrix[i][j]); } printf("\n"); } } // 广播维度信息 int dims[2]; if (rank == 0) { dims[0] = rows; dims[1] = cols; } MPI_Bcast(dims, 2, MPI_INT, 0, MPI_COMM_WORLD); rows = dims[0]; cols = dims[1]; // 非根进程分配内存 if (rank != 0) { matrix = (int**)malloc(rows * sizeof(int*)); for (int i = 0; i < rows; i++) { matrix[i] = (int*)malloc(cols * sizeof(int)); } } // 逐行广播(二维数组在内存中不连续) for (int i = 0; i < rows; i++) { MPI_Bcast(matrix[i], cols, MPI_INT, 0, MPI_COMM_WORLD); } // 验证结果 printf("Rank %d: matrix[1][2] = %d\n", rank, matrix[1][2]); // 清理内存 for (int i = 0; i < rows; i++) { free(matrix[i]); } free(matrix); MPI_Finalize(); return 0; }

4.使用派生数据类型广播连续内存块

#include <stdio.h> #include <stdlib.h> #include <mpi.h> typedef struct { int id; double value; char name[20]; } Data; int main(int argc, char** argv) { MPI_Init(&argc, &argv); int rank; MPI_Comm_rank(MPI_COMM_WORLD, &rank); int count = 5; Data* data_array = NULL; // 创建派生数据类型 MPI_Datatype MPI_DATA_TYPE; int blocklengths[3] = {1, 1, 20}; MPI_Aint displacements[3]; MPI_Datatype types[3] = {MPI_INT, MPI_DOUBLE, MPI_CHAR}; Data dummy; MPI_Get_address(&dummy.id, &displacements[0]); MPI_Get_address(&dummy.value, &displacements[1]); MPI_Get_address(&dummy.name, &displacements[2]); // 计算相对位移 for (int i = 2; i >= 0; i--) { displacements[i] -= displacements[0]; } MPI_Type_create_struct(3, blocklengths, displacements, types, &MPI_DATA_TYPE); MPI_Type_commit(&MPI_DATA_TYPE); // 根进程初始化数据 if (rank == 0) { data_array = (Data*)malloc(count * sizeof(Data)); for (int i = 0; i < count; i++) { data_array[i].id = i; data_array[i].value = i * 1.5; sprintf(data_array[i].name, "Item_%d", i); } } else { data_array = (Data*)malloc(count * sizeof(Data)); } // 广播数据 MPI_Bcast(data_array, count, MPI_DATA_TYPE, 0, MPI_COMM_WORLD); // 验证结果 printf("Rank %d: Data[2] = {id:%d, value:%.2f, name:%s}\n", rank, data_array[2].id, data_array[2].value, data_array[2].name); free(data_array); MPI_Type_free(&MPI_DATA_TYPE); MPI_Finalize(); return 0; }

5.编译和运行

# 编译 mpicc -o broadcast_array broadcast_array.c # 运行(使用4个进程) mpirun -np 4 ./broadcast_array # 或指定机器文件 mpirun -np 4 -machinefile hosts.txt ./broadcast_array

6.MPI 广播的知识点总结

  1. MPI_Bcast 参数

    int MPI_Bcast(void *buffer, // 数据缓冲区 int count, // 元素数量 MPI_Datatype datatype, // 数据类型 int root, // 根进程rank MPI_Comm comm) // 通信域
  2. 广播步骤

    • 根进程准备数据

    • 非根进程分配内存

    • 广播长度信息(如果需要)

    • 广播数据本身

  3. 内存管理

    • 所有进程都需要为接收的数据分配内存

    • 广播前内存必须已分配

    • 广播后所有进程的数据完全相同

  4. 性能考虑

    • 大数组广播可能成为性能瓶颈

    • 考虑使用 scatter/gather 或特定模式

    • 对于非常大的数据,可能需要分段广播

  5. 错误处理

    • 检查内存分配是否成功

    • 验证广播返回值

    • 确保所有进程使用相同的参数

这个示例展示了 MPI 中广播数组的基本用法,可以根据实际需求进行调整和优化。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/9 14:13:25

.NET开发-PDF处理不重复造轮子!PDFPatcher省去80%代码,聚焦核心业务

前言PDF格式因稳定性强、跨平台兼容的特性&#xff0c;被广泛应用于办公文档传输、资料存档、合同流转等各类场景&#xff0c;但其只读属性也带来了诸多使用局限——常规操作中&#xff0c;用户在编辑页面、调整结构、解除权限限制、合并拆分文档等需求上往往束手无策&#xff…

作者头像 李华
网站建设 2026/5/9 16:59:02

C# 实现简版 Claude Code | Bash 就是一切(1)

❝该系列文章基于 github.com/shareAI-lab/learn-claude-code 写就&#xff0c;该仓库以大道至简的风格剖析了Claude Code的核心原理&#xff0c;值得大家学习。由于该仓库是基于Python语言&#xff0c;为方便.NET开发者学习&#xff0c;我已经将代码基于.NET 10的dotnet file …

作者头像 李华
网站建设 2026/5/10 11:17:55

基于大数据的热门旅游景点推荐系统(源码+lw+部署文档+讲解等)

课题介绍 本课题旨在设计并实现一套基于大数据的热门旅游景点推荐系统&#xff0c;解决当前旅游出行中景点信息杂乱、推荐精准度低、用户偏好匹配不足、出行决策效率低等问题&#xff0c;适配游客出行规划、旅游平台运营及景区宣传推广的核心需求。系统融合大数据技术与相关开发…

作者头像 李华
网站建设 2026/5/9 18:49:38

基于大数据的专业智能导学系统的设计与实现(源码+lw+部署文档+讲解等)

课题介绍本课题旨在设计并实现一套基于大数据的专业智能导学系统&#xff0c;解决当前专业教学中导学针对性不足、学习路径模糊、学情分析滞后、个性化辅导缺失等问题&#xff0c;适配高校专业教学、学生自主学习及教师导学辅导的核心需求。系统融合大数据技术与相关开发框架&a…

作者头像 李华
网站建设 2026/5/9 13:57:46

不踩雷!领军级的AI论文工具 —— 千笔·专业学术智能体

你是否曾为论文选题而苦恼&#xff1f;是否在深夜面对空白文档无从下笔&#xff1f;是否反复修改却仍对表达不满意&#xff1f;论文写作不仅是学术能力的体现&#xff0c;更是时间与精力的消耗。对于继续教育的学生来说&#xff0c;如何高效完成高质量的论文&#xff0c;是毕业…

作者头像 李华