0

0

c++中std::forward_iterator和std::bidirectional_iterator有什么区别? (迭代器概念)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-12 12:30:02

|

649人浏览过

|

来源于php中文网

原创

std::forward_iterator 是单向可读写迭代器,支持++、*、==、!=及复制比较,但不支持--或随机访问;std::bidirectional_iterator在此基础上增加--操作,支持双向遍历,如std::list::iterator;二者为概念分层关系,非类型别名继承。

c++中std::forward_iterator和std::bidirectional_iterator有什么区别? (迭代器概念)

std::forward_iterator 是单向可读写的迭代器概念

它要求迭代器支持 ++it(前缀递增)、it++(后缀递增)、*it(解引用)、==!=,且必须满足“可复制构造/赋值”和“可比较相等性”的语义。典型代表是 std::forward_list::iterator 和输入流迭代器(如 std::istream_iterator)。

关键限制:不支持 --itit--;不能倒退;一旦递增,原值不可恢复(除非额外保存)。

  • 只保证单向遍历:从 begin 到 end,不能回头
  • 不要求支持 -> 操作符(但多数实现提供)
  • 不要求支持 it + nit - n 或随机访问
  • 对底层容器无所有权或修改要求;只承诺“能读、能走、能判等”

std::bidirectional_iterator 在 forward_iterator 基础上增加反向能力

它继承自 std::forward_iterator,并额外要求支持 --itit--。这意味着你可以前后移动,适合需要双向扫描的算法,比如 std::reversestd::list::sort

典型代表是 std::list::iteratorstd::set::iterator,以及 C++20 中的 std::deque::iterator(注意:deque 迭代器其实是 random_access_iterator,但它也满足 bidirectional)。

立即学习C++免费学习笔记(深入)”;

  • 必须能“来得去得”:任意位置可前进一格,也可后退一格
  • 仍不支持算术运算(如 it + 5)或下标访问(it[3]
  • 所有 bidirectional_iterator 都是 forward_iterator,但反之不成立
  • 某些操作(如 std::prev(it))在 bidirectional 上是 O(1),在 forward 上是 O(n)

为什么 std::vector::iterator 不是 bidirectional_iterator?

它其实是 std::random_access_iterator —— 这是一个更强的概念,已隐含 bidirectional 和 forward 的所有要求。C++ 标准库中迭代器概念是分层的:input_iteratorforward_iteratorbidirectional_iteratorrandom_access_iteratorcontiguous_iterator

Packify
Packify

Packify 是一个创新的AI包装设计工具

下载

所以 std::vector::iterator--、能 +、能 [],但它“属于更高阶概念”,编译器不会把它降级为 bidirectional_iterator 类型别名 —— 类型系统靠 concept 约束,而非别名继承。

  • static_assert(std::bidirectional_iterator<:vector>::iterator>) 会通过(因为满足约束)
  • std::vector::iterator 的类型定义本身不是 typedefbidirectional_iterator
  • 真正区分行为的是 concept 检查,不是 typedef 名字

实际写泛型代码时怎么选 concept?

取决于你要做的操作。如果你只遍历一次、不需要回头,用 std::forward_iterator 更宽泛,能接受更多容器(如 std::forward_list)。如果要 reverse 或双指针扫描(如 partition),就必须提升到 std::bidirectional_iterator

错误示例:给一个只接受 bidirectional_iterator 的函数传入 std::forward_list::iterator,编译失败,报错类似 no match for 'operator--'

template
It find_last(It first, It last, const auto& val) {
    while (first != last) {
        --last; // ← 这里要求 --last 合法
        if (*last == val) return last;
    }
    return last;
}

这个函数对 std::list 有效,但无法用于 std::forward_list —— 即便你手动重写成从头扫,逻辑也不同。概念不是“能不能凑合”,而是“接口契约是否完整”。

最容易被忽略的一点:concept 判断的是表达式有效性 + 语义要求(如可比较、可复制),不是看类有没有叫 operator-- 的成员函数。有些自定义迭代器重载了 -- 但不满足可交换性或可复制性,依然不算 bidirectional_iterator

相关专题

更多
sort排序函数用法
sort排序函数用法

sort排序函数的用法:1、对列表进行排序,默认情况下,sort函数按升序排序,因此最终输出的结果是按从小到大的顺序排列的;2、对元组进行排序,默认情况下,sort函数按元素的大小进行排序,因此最终输出的结果是按从小到大的顺序排列的;3、对字典进行排序,由于字典是无序的,因此排序后的结果仍然是原来的字典,使用一个lambda表达式作为key参数的值,用于指定排序的依据。

384

2023.09.04

typedef和define区别
typedef和define区别

typedef和define区别在类型检查、作用范围、可读性、错误处理和内存占用等。本专题为大家提供typedef和define相关的文章、下载、课程内容,供大家免费下载体验。

107

2023.09.26

c语言typedef的用法
c语言typedef的用法

c语言typedef的用法有定义基本类型别名、定义结构体别名、定义指针类型别名、定义枚举类型别名、定义数组类型别名等。本专题为大家提供typedef相关的文章、下载、课程内容,供大家免费下载体验。

96

2023.09.26

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1011

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

60

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

370

2025.12.29

页面置换算法
页面置换算法

页面置换算法是操作系统中用来决定在内存中哪些页面应该被换出以便为新的页面提供空间的算法。本专题为大家提供页面置换算法的相关文章,大家可以免费体验。

399

2023.08.14

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

78

2026.01.09

c++框架学习教程汇总
c++框架学习教程汇总

本专题整合了c++框架学习教程汇总,阅读专题下面的文章了解更多详细内容。

46

2026.01.09

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
SQL 教程
SQL 教程

共61课时 | 3.4万人学习

C# 教程
C# 教程

共94课时 | 6.5万人学习

C 教程
C 教程

共75课时 | 4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号