news 2026/4/3 4:45: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/3/30 16:53:45

工业机器人轨迹怎么来?这个 C# 工具从 DXF 自动生成

前言工业自动化日益普及&#xff0c;如何将设计师手中的 CAD 图纸高效、准确地转化为机器人可以执行的运动路径&#xff0c;成为连接创意与制造的关键环节。很多小型产线或创客项目往往缺乏成熟的 CAM&#xff08;计算机辅助制造&#xff09;系统&#xff0c;导致这一转换过程既…

作者头像 李华
网站建设 2026/4/2 10:34:22

基于大数据的美妆产品网络评价的数据采集与分析(源码+lw+部署文档+讲解等)

课题介绍 本课题旨在设计并实现一套基于大数据的美妆产品网络评价的数据采集与分析系统&#xff0c;解决当前美妆行业中网络评价分散杂乱、采集效率低、情感倾向判断不准、评价价值挖掘不足等问题&#xff0c;适配美妆企业市场调研、产品优化及消费者购买决策的核心需求。系统融…

作者头像 李华
网站建设 2026/3/27 0:58:26

调查:超1/4游戏开发者近两年遭解雇,近一半仍未找到新工作;阿里自研AI芯片“真武”亮相;华为手机全系降价,最高达4000元 | 极客头条

「极客头条」—— 技术人员的新闻圈&#xff01; CSDN 的读者朋友们好&#xff0c;「极客头条」来啦&#xff0c;快来看今天都有哪些值得我们技术人关注的重要新闻吧。&#xff08;投稿或寻求报道&#xff1a;zhanghycsdn.net&#xff09; 整理 | 苏宓 出品 | CSDN&#xff08;…

作者头像 李华
网站建设 2026/4/1 21:08:53

Servlet入门保姆级教程:从0到1搞定Tomcat与Web开发

作为Java后端新手&#xff0c;刚接触Servlet是不是总被一堆“高大上”的概念绕晕&#xff1f;Tomcat是啥&#xff1f;CS/BS架构有啥区别&#xff1f;HTTP协议到底在聊啥&#xff1f;别急&#xff01;这篇文章用最接地气的话实战案例&#xff0c;带你轻松拿捏Servlet入门核心知识…

作者头像 李华
网站建设 2026/3/30 22:45:34

基于SpringBoot和Vue的物流管理系统(源码+lw+部署文档+讲解等)

课题介绍 本课题旨在设计并实现一套基于SpringBoot和Vue的物流管理系统&#xff0c;解决当前物流运营中订单管理混乱、货物追踪不便、仓储与运输协同低效、数据统计滞后等问题&#xff0c;适配中小型物流企业规范化、信息化运营的核心需求。系统采用前后端分离架构&#xff0c;…

作者头像 李华
网站建设 2026/3/16 5:48:07

基于SpringBoot的校园便利平台毕设

博主介绍&#xff1a;✌ 专注于Java,python,✌关注✌私信我✌具体的问题&#xff0c;我会尽力帮助你。一、研究目的本研究旨在设计并实现一个基于SpringBoot框架的校园便利平台&#xff0c;以满足现代校园生活中学生、教职工及访客对于便捷服务需求。具体研究目的如下&#xff…

作者头像 李华