首页 > 编程语言 >C++23:多维视图(std::mdspan)

C++23:多维视图(std::mdspan)

时间:2023-10-30 13:02:28浏览次数:46  
标签:std myVec 23 视图 C++ 多维 mdspan include


C++23:多维视图(std::mdspan)

介绍

在 C++23 中,std::mdspan 是一个非拥有的多维视图,用于表示连续对象序列。这个连续对象序列可以是一个简单的 C 数组、带有大小的指针、std::arraystd::vectorstd::string

这种多维视图通常被称为多维数组。

多维数组的形状由其维数(也称为秩)和每个维度的大小(也称为扩展)决定。std::mdspan 的大小是所有非零维度的大小的乘积。你可以使用多维索引运算符 [] 来访问 std::mdspan 的元素。

每个 std::mdspan 的维度可以有静态或动态的扩展。静态扩展意味着其长度在编译时指定;动态扩展意味着其长度在运行时指定。

定义

template<
    class T,
    class Extents,
    class LayoutPolicy = std::layout_right,
    class AccessorPolicy = std::default_accessor<T>
> class mdspan;
  • T: 连续对象序列的类型。
  • Extents: 指定维数及其大小;每个维度可以有静态或动态的扩展。
  • LayoutPolicy: 指定用于访问底层内存的布局策略。
  • AccessorPolicy: 指定如何引用底层元素。

由于 C++17 中的类模板参数推导(CTAD),编译器通常可以自动从初始化器的类型推导出模板参数。

示例

使用动态扩展

#include <mdspan>
#include <iostream>
#include <vector>

int main() {
    
    std::vector myVec{1, 2, 3, 4, 5, 6, 7, 8};          // (1)

    std::mdspan m{myVec.data(), 2, 4};                  // (2)
    std::cout << "m.rank(): " << m.rank() << '\n';      // (4)

    for (std::size_t i = 0; i < m.extent(0); ++i) {     // (6)
        for (std::size_t j = 0; j < m.extent(1); ++j) { // (7)
            std::cout << m[i, j] << ' ';                // (8)
        }
        std::cout << '\n';
    }

    std::cout << '\n';

    std::mdspan m2{myVec.data(), 4, 2};                 // (3)
    std::cout << "m2.rank(): " << m2.rank() << '\n';    // (5)

    for (std::size_t i = 0; i < m2.extent(0); ++i) {
        for (std::size_t j = 0; j < m2.extent(1); ++j) {
        std::cout << m2[i, j] << ' ';  
    }
    std::cout << '\n';
  }

}

使用静态和动态扩展

#include <mdspan>
#include <iostream>
#include <string>
#include <vector>
#include <tuple>

int main() {
    
    std::vector myVec{1, 2, 3, 4, 5, 6, 7, 8};

    std::mdspan<int, std::extents<std::size_t, 2, 4>> m{myVec.data()}; // (1)
    std::cout << "m.rank(): " << m.rank() << '\n';

    for (std::size_t i = 0; i < m.extent(0); ++i) {
        for (std::size_t j = 0; j < m.extent(1); ++j) {
            std::cout << m[i, j] << ' ';  
        }
        std::cout << '\n';
    }

    std::cout << '\n';

    std::mdspan<int, std::extents<std::size_t, std::dynamic_extent, 
                std::dynamic_extent>> m2{myVec.data(), 4, 2};          // (2)
    std::cout << "m2.rank(): " << m2.rank() << '\n';

    for (std::size_t i = 0; i < m2.extent(0); ++i) {
        for (std::size_t j = 0; j < m2.extent(1); ++j) {
        std::cout << m2[i, j] << ' ';  
    }
    std::cout << '\n';
  }

   std::cout << '\n';

}

布局策略

std::mdspan 允许您指定用于访问底层内存的布局策略。默认情况下,使用 std::layout_right(C、C++ 或 Python 风格),但您也可以指定 std::layout_left(Fortran 或 MATLAB 风格)。

C++23:多维视图(std::mdspan)_布局策略

使用布局策略std::mdspanstd::layout_right遍历两个std::layout_left可以看出差异。

#include <mdspan>
#include <iostream>
#include <vector>

int main() {
    
    std::vector myVec{1, 2, 3, 4, 5, 6, 7, 8};

    std::mdspan<int, std::extents<std::size_t,      // (1)
         std::dynamic_extent, std::dynamic_extent>, 
         std::layout_right> m{myVec.data(), 4, 2};
    std::cout << "m.rank(): " << m.rank() << '\n';

    for (std::size_t i = 0; i < m.extent(0); ++i) {
        for (std::size_t j = 0; j < m.extent(1); ++j) {
            std::cout << m[i, j] << ' ';  
        }
        std::cout << '\n';
    }

    std::cout << '\n';

    std::mdspan<int, std::extents<std::size_t,     // (2)
         std::dynamic_extent, std::dynamic_extent>, 
         std::layout_left> m2{myVec.data(), 4, 2};
    std::cout << "m2.rank(): " << m2.rank() << '\n';

    for (std::size_t i = 0; i < m2.extent(0); ++i) {
        for (std::size_t j = 0; j < m2.extent(1); ++j) {
        std::cout << m2[i, j] << ' ';  
    }
    std::cout << '\n';
  }

}

C++23:多维视图(std::mdspan)_布局策略_02

接口概览

  • md[ind]: 访问第 ind 个元素。
  • md.size: 返回多维数组的大小。
  • md.rank: 返回多维数组的维数。
  • md.extents(i): 返回第 i 个维度的大小。
  • md.data_handle: 返回指向连续元素序列的指针。

参考:MC++ BLOG – Der Blog für Modernes C++ von Rainer Grimm (modernescpp.com)


标签:std,myVec,23,视图,C++,多维,mdspan,include
From: https://blog.51cto.com/u_16062556/8087706

相关文章

  • 【专题】腾讯数字孪生云白皮书2023报告PDF合集分享(附原数据表)
    原文链接:https://tecdat.cn/?p=34041本次报告合集分为数字孪生综述、技术架构建设、核心技术分享、新型技术成果展示以及重点行业应用五大内容版块。从数字孪生应用建设路径的角度出发,着重提出了“数智视融合,虚实人联动”的观点,并提供数字孪生应用技术的参考。同时,本报告合集还完......
  • 【专题】数字孪生世界白皮书(2023)报告PDF合集分享(附原数据表)
    原文链接:https://tecdat.cn/?p=34041本次报告合集分为数字孪生综述、技术架构建设、核心技术分享、新型技术成果展示以及重点行业应用五大内容版块。从数字孪生应用建设路径的角度出发,着重提出了“数智视融合,虚实人联动”的观点,并提供数字孪生应用技术的参考。同时,本报告合集还完......
  • 【专题】数字孪生城市创新应用场景研究报告(2023)报告PDF合集分享(附原数据表)
    原文链接:https://tecdat.cn/?p=34041本次报告合集分为数字孪生综述、技术架构建设、核心技术分享、新型技术成果展示以及重点行业应用五大内容版块。从数字孪生应用建设路径的角度出发,着重提出了“数智视融合,虚实人联动”的观点,并提供数字孪生应用技术的参考。同时,本报告合集还完......
  • 【专题】2023年中国数字孪生城市行业研究报告PDF合集分享(附原数据表)
    原文链接:https://tecdat.cn/?p=34041本次报告合集分为数字孪生综述、技术架构建设、核心技术分享、新型技术成果展示以及重点行业应用五大内容版块。从数字孪生应用建设路径的角度出发,着重提出了“数智视融合,虚实人联动”的观点,并提供数字孪生应用技术的参考。同时,本报告合集还完......
  • 又是1024诶,这是20231024,总结事业编考试记得写名字
    最近也是感慨颇多。总之我想说或者建议:1、考试一般要写名字都会在明显的地方2、对于不常见考试,是否监考可以提醒考生在哪里写名字?(至少提醒哪些答题卡要写名字?)3、对于考后是否可以写名字,监考帮考生写名字也可以不。在老家参与了一次事业单位考试,考的公基和写作,然后最近一直在备考国......
  • 2023了,考华为认证还有用吗?
    华为作为全球领先的通信设备供应商,其网络工程师认证具有极高的含金量。通过考取华为网络工程师认证,意味着您具备了扎实的网络基础知识、熟练的实践操作技能以及较高的职业素养,这对于您在IT行业中的职业发展将起到积极的推动作用。那么2023年考华为网络工程师认证还有用吗?下面我们就......
  • 安全认证 | 2023年CISP考试报名条件
    CISP是国内网络信息安全领域唯一认可、专注于培养与考核高级应用安全技术人才的渗透测试证书。那CISP报名需要满足什么条件么,让我们一起来看一下。012023年CISP考试报名条件及CISP认证要求介绍首先在校生不能报考CISP证书,如果可以,可以报考NISP证书。CISP报考条件主要从以下几点进......
  • 23/10/29 模拟赛总结
    时间安排7:35-8:20直接开T1,发现不会做。8:20-8:40把T2T3T4都看了,T2和T1一样是我必不可能会的人类智慧题,T3看上去就很劝退,T4是我喜欢的树上问题,直接倒序开题。8:40-10:20想了T4,得到了一个\(O(n^2)\)的暴力,高达40分,直接开写。写完之后过不了第二个样例,发现假......
  • 10.23~10.29
    补题补了Mea的Math2反演内容。学习了一下树分块的模板。补了部分Hanghang的dp优化。补了一点基础DS、基础dp。比赛打了一场lxsround和北大附联考,感觉发挥不错(希望NOIP有这个状态),但是都有挂分。lxsround第四题写挂了,100->40。北大附联考第三题写挂了,100->......
  • 周藤2023游记
    周藤的2023游记,大家不喜欢别喷周藤的2023游记,大家不喜欢别喷周藤的2023游记,大家不喜欢别喷周藤的2023游记,大家不喜欢别喷周藤的2023游记,大家不喜欢别喷周藤的2023游记,大家不喜欢别喷周藤的2023游记,大家不喜欢别喷周藤的2023游记,大家不喜欢别喷周藤的2023游记,大家不喜欢别喷......