news 2026/5/30 15:48:04

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/30 0:16:50

三步快速摸清陌生研究领域,轻松搞定文献调研

对于研究生和博士生而言&#xff0c;科研生涯中往往存在着几个令人焦虑的“至暗时刻”&#xff1a;研究生开题、博士确定研究方向&#xff0c;以及申请基金前的立项论证。这些阶段的核心任务惊人的一致——你需要在一个有限的时间内&#xff0c;快速、全面、深入地了解某个陌生…

作者头像 李华
网站建设 2026/5/29 14:21:20

Maven踩坑指南:依赖冲突专治不服,范围聚合玩明白!

谁说Maven只有枯燥的配置&#xff1f;今天咱用“唠嗑式”教学&#xff0c;把依赖冲突、依赖范围、项目聚合拆解得明明白白&#xff0c;新手也能秒懂&#x1f61c; 一、依赖冲突&#xff1a;Maven版“版本打架”怎么破&#xff1f; 咱先还原一个真实场景&#xff1a;你开开心心…

作者头像 李华
网站建设 2026/5/27 1:41:16

基于Springboot+Vue的绥大学生学习平台管理系统源码文档部署文档代码讲解等

课题介绍 本课题旨在设计并实现一套基于SpringBootVue的绥大学生学习平台管理系统&#xff0c;解决绥大学生学习过程中资源分散、学习进度难追踪、师生互动不便及管理员管控低效等问题&#xff0c;适配绥大教学管理与学生自主学习的核心需求。系统采用前后端分离架构&#xff0…

作者头像 李华
网站建设 2026/5/30 12:55:52

AI原生应用领域思维树的创新模式探讨

AI原生应用领域思维树的创新模式探讨 关键词&#xff1a;AI原生应用、思维树&#xff08;Tree of Thoughts, ToT&#xff09;、大语言模型&#xff08;LLM&#xff09;、多步推理、生成式AI 摘要&#xff1a;随着生成式AI技术的爆发&#xff0c;“AI原生应用”&#xff08;AI-N…

作者头像 李华
网站建设 2026/5/25 1:54:09

游戏在 HarmonyOS 上如何“活”?

子玥酱 &#xff08;掘金 / 知乎 / CSDN / 简书 同名&#xff09; 大家好&#xff0c;我是 子玥酱&#xff0c;一名长期深耕在一线的前端程序媛 &#x1f469;‍&#x1f4bb;。曾就职于多家知名互联网大厂&#xff0c;目前在某国企负责前端软件研发相关工作&#xff0c;主要聚…

作者头像 李华
网站建设 2026/5/28 21:56:13

基于multisim的可控直流稳压电源的设计与仿真

具体参数要求:输入电压:220V:输出电压:1.25-15V可调直流电压;输出电流:最大电流为1.5A:保护电路:过流保护、短路保护。 仿真图&#xff1a; 仿真演示与文件下载&#xff1a;基于multisim的可控直流稳压电源的设计与仿真演示视频_哔哩哔哩_bilibili

作者头像 李华