当前位置: 首页 > ops >正文

C++23:ranges::iota、ranges::shift_left和ranges::shift_right详解

文章目录

    • 引言
    • `ranges::iota`
      • 定义与功能
      • 使用场景
      • 代码示例
    • `ranges::shift_left`
      • 定义与功能
      • 使用场景
      • 代码示例
    • `ranges::shift_right`
      • 定义与功能
      • 使用场景
      • 代码示例
    • 总结

引言

C++23作为C++编程语言的一个重要版本,为开发者带来了许多新的特性和改进。其中,ranges::iotaranges::shift_leftranges::shift_right这三个无约束算法的范围化版本(P2440R1)为处理序列数据提供了更加便捷和高效的方式。本文将详细介绍这三个算法的定义、功能、使用场景以及代码示例。

ranges::iota

定义与功能

ranges::iotastd::iota的范围化版本,它在C++11中就已经存在,而在C++23中得到了进一步的扩展和优化。std::iota函数定义在<numeric>头文件中,用于将一个连续递增的值填充到指定的范围[first, last)中,起始值为value,并重复执行++value操作。其函数原型如下:

#include <numeric>
template < class ForwardIt, class T >
void iota( ForwardIt first, ForwardIt last, T value );
(since C++11)
(constexpr since C++20)

ranges::iota的功能与之类似,但它更符合现代C++的范围编程范式,能够更方便地与其他范围算法和视图结合使用。

使用场景

ranges::iota常用于初始化数组或容器,为其赋予连续递增的值。例如,我们可以使用它来初始化一个包含连续整数的向量:

#include <iostream>
#include <vector>
#include <numeric>
#include <ranges>int main() {std::vector<int> vec(10);std::ranges::iota(vec, 1);for (auto num : vec) {std::cout << num << " ";}std::cout << std::endl;return 0;
}

在上述代码中,std::ranges::iota(vec, 1)将从1开始的连续整数填充到vec向量中。

代码示例

以下是一个更复杂的示例,展示了如何使用ranges::iota与其他范围视图结合:

#include <iostream>
#include <ranges>
#include <algorithm>int main() {auto numbers = std::views::iota(1, 11);auto squared = numbers | std::views::transform([](int x) { return x * x; });std::ranges::for_each(squared, [](int x) { std::cout << x << " "; });std::cout << std::endl;return 0;
}

在这个示例中,std::views::iota(1, 11)生成了一个从1到10的整数范围,然后使用std::views::transform视图将每个数平方,最后使用std::ranges::for_each遍历并输出结果。

ranges::shift_left

定义与功能

ranges::shift_left是一个用于将范围中的元素向左移动指定位置的算法。它定义在<algorithm>头文件中,函数原型如下:

#include <algorithm>
template< std::permutable I, std::sentinel_for<I> S >
constexpr ranges::subrange<I>shift_left( I first, S last, std::iter_difference_t<I> n ); (1) (since C++23)
template< ranges::forward_range R >
requires std::permutable<ranges::iterator_t<R>>
constexpr ranges::borrowed_subrange_t<R>shift_left( R&& r, ranges::range_difference_t<R> n ); (2) (since C++23)

该算法将范围[first, last)r中的元素向左移动n个位置。如果n == 0 || n >= last - first,则没有任何效果;如果n < 0,行为是未定义的。否则,对于[0, last - first - n)中的每个整数i,将原本位于位置first + n + i的元素移动到位置first + i

使用场景

ranges::shift_left常用于需要对序列进行循环移位的场景,例如在处理环形缓冲区或循环队列时。

代码示例

#include <iostream>
#include <vector>
#include <algorithm>
#include <ranges>int main() {std::vector<int> numbers = {1, 2, 3, 4, 5};int shiftAmount = 2;std::ranges::shift_left(numbers, shiftAmount);for (const auto& number : numbers) {std::cout << number << " ";}std::cout << std::endl;return 0;
}

在这个示例中,std::ranges::shift_left(numbers, 2)numbers向量中的元素向左移动了2个位置。

ranges::shift_right

定义与功能

ranges::shift_rightranges::shift_left相反,它用于将范围中的元素向右移动指定位置。其函数原型如下:

#include <algorithm>
template< std::permutable I, std::sentinel_for<I> S >
constexpr ranges::subrange<I>shift_right( I first, S last, std::iter_difference_t<I> n ); (3) (since C++23)
template< ranges::forward_range R >
requires std::permutable<ranges::iterator_t<R>>
constexpr ranges::borrowed_subrange_t<R>shift_right( R&& r, ranges::range_difference_t<R> n ); (4) (since C++23)

该算法将范围[first, last)r中的元素向右移动n个位置。如果n == 0 || n >= last - first,则没有任何效果;如果n < 0,行为是未定义的。否则,对于[0, last - first - n)中的每个整数i,将原本位于位置first + i的元素移动到位置first + n + i

使用场景

ranges::shift_right同样适用于需要对序列进行循环移位的场景,只是方向相反。

代码示例

#include <iostream>
#include <vector>
#include <ranges>int main() {std::vector<int> numbers = {1, 2, 3, 4, 5};std::ranges::shift_right(numbers, 2);for (const auto& number : numbers) {std::cout << number << " ";}std::cout << std::endl;return 0;
}

在这个示例中,std::ranges::shift_right(numbers, 2)numbers向量中的元素向右移动了2个位置。

总结

C++23中的ranges::iotaranges::shift_leftranges::shift_right这三个无约束算法的范围化版本为开发者提供了更加强大、便捷和高效的序列处理能力。它们不仅简化了代码的编写,还提高了代码的可读性和可维护性。通过合理使用这些算法,我们可以更加轻松地处理各种序列数据,提升程序的性能和质量。

http://www.xdnf.cn/news/6536.html

相关文章:

  • 在 Neo4j 中实现向量化存储:从文本到高效语义搜索
  • 【爬虫】DrissionPage-4
  • h5,原生html,echarts关系网实现
  • 降低学习成本,1 天掌握 Java 开发核心技能
  • 【带文档】网上点餐系统 springboot + vue 全栈项目实战(源码+数据库+万字说明文档)
  • 《AI大模型应知应会100篇》第66篇:用大模型 + 向量数据库构建你的个性化知识库系统(附实战代码)
  • FPGA: XILINX Kintex 7系列器件的架构
  • 广州附医华南医院MDT团队新技术赋能「睡眠障碍治疗」:告别单一、反复、不对症,实现精准快速起效!
  • 架构选择、区别
  • C++动态内存分配
  • MySQL高可用
  • GEE计算 RSEI(遥感生态指数)
  • 无损耗协议:PROFINET和EtherNet IP网关的高效安装指南
  • SSH主机密钥验证失败:全面解决方案与技术手册
  • 2025-5-16Vue3快速上手
  • 填孔即可靠:猎板PCB如何用树脂塞孔重构高速电路设计规则
  • 从前序与中序遍历序列构造二叉树(中等)
  • 【linux】Web服务—搭建nginx+ssl的加密认证web服务器
  • Ubuntu快速安装Python3.11及多版本管理
  • 项目版本管理和Git分支管理方案
  • Android 中 显示 PDF 文件内容(AndroidPdfViewer 库)
  • 计算机图形学编程(使用OpenGL和C++)(第2版)学习笔记 10.增强表面细节(二)法线贴图
  • SpringCloud微服务开发与实战
  • 官方 Elasticsearch SQL NLPChina Elasticsearch SQL
  • [特殊字符][特殊字符]知识库PHP版 | ChatMoneyAI宝塔面板Docker多部署
  • Java EE初阶——wait 和 notify
  • CentOS高手之路:从进阶实战到企业级优化
  • 维智定位 Android 定位 SDK
  • 网站运维基础 | 2. cms介绍及wordpress的搭建
  • 物联网中的WiFi模式解析:AP、STA与混合模式