C++标准泛型算法库的函数为什么设计成不能改变容器大小?


        相信蔡老师应该是看了Primer之后产生的这样的疑问,但首先注意的是C++泛型算法的操作对象从来就不是容器,更别提为什么不能改变容器的大小。泛型算法,无论是只读的(比如find,accumulate)还是可写的(sort,fill),其输入参数都是迭代器,也就是说泛型算法眼中只有迭代器,它是看不到容器的。泛型算法有什么样的特性取决于给它提供的迭代器有什么能力。迭代器按照能力大小可分为输入/输出迭代器、向前迭代器、双向迭代器、随机访问迭代器。这些迭代器能力由低到高,高能力迭代器能够实现低能力迭代器的所有功能。那为什么不将泛型算法直接作用于容器呢?泛型算法,顾名思义,就是一种高层算法,它将各种容器和它们的操作进行了隔离,通过抽象出迭代器这一概念,实现了对所有容器的统一操作。比如array、deque、string、vector这些容器都提供了最高能力的随机迭代器,所以可以对它们进行统一的sort算法,如果没有迭代器这一中间层,算法一定会与底层容器耦合,这就失去了泛型算法的意义,也就是说必须要为每种容器设计独立的算法。现在可以大致确定,迭代器这一概念的重要性,所以我们可以理解泛型算法不能直接改变容器大小,但是如果容器提供的迭代器的能力足够,是可以间接改变大小的。比如:
remove(beg, end, val);
unique(beg, end);
        这类泛型算法最低要求前向迭代器,而恰好array、deque、string、vector、list、forward_list都提供了这种迭代器,所以这些算法可以间接改变这类容器的大小。为什么back_inserter这样的函数能在不违反“不改变容器大小”的规范上在容器尾添加元素?back_insert_iterator、front_insert_iterator、insert_iterator这类的插入迭代器(其实是一种适配器)是独立于容器的,它们属于std空间,而容器迭代器属于自身的子空间中,所以它们的用法是有区别的(注意它们的声明方式):
vector<int> vec;
back_insert_iterator<vector<int>> it = back_inserter(vec);
vector<int>::iterator it = vec.begin();

        插入迭代器是一种适配器,back_insert_iterator必须作用于支持push_back的容器,front_insert_iterator必须作用于支持push_front的容器,仅仅是对这两个方法的封装。所以最好把它们和普通的容器迭代器区分开来。那为什么要把push_back和push_front进行封装呢?原因有两个,一是用着更方便,二是扩充了迭代器的能力,因为容器的迭代器最高也就是随机访问的能力,但是你不能通过迭代器扩充容器。~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~评论里说是不是可以设计删除迭代器?可以当然是可以,迭代器想怎么设计就怎么设计,比如可以像插入迭代器那样通过封装容器的erase方法来构造删除迭代器,但是有什么意义呢?因为通过普通的迭代器已经可以实现删除操作了。比如之前提到的:
remove(beg, end, val)
unique(beg, end)


声明:
作者:Aloys寒风
链接:https://www.zhihu.com/question/28495347/answer/42180143
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
  • 3
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Haskell的容器类型和C的泛型有一些显著的差异: 1. 类型系统:Haskell是一门强类型的语言,因此在定义容器类型时,必须明确指定容器中存储的数据类型。而C的泛型类型则可以在编译时确定。 2. 可读性:Haskell的容器类型在类型系统的基础上,提供了更好的可读性。这使得代码更加可靠,减少了代码错误。 3. 类型安全:Haskell的类型系统提供了更强的类型安全性,从而减少了类型相关的错误。 4. 抽象性:Haskell的容器类型更具抽象性,可以通过使用函数式编程的技巧实现更复杂的数据结构。 5. 函数式特性:Haskell是一门函数式语言,因此它的容器类型与函数式编程的理念紧密相关。在Haskell中,容器类型与函数结合使用可以实现高效、简洁和易于理解的代码。 总之,Haskell的容器类型与C的泛型有着显著的差异,但它们都是有效的数据结构实现方式。 ### 回答2: Haskell的容器类型和C的泛型有以下几点区别: 1. 编程语言特性:Haskell是一种纯函数式编程语言,而C是一种命令式编程语言。Haskell的容器类型是通过类型系统来保证数据的一致性和不可变性的,而C的泛型是通过预处理器宏和指针转换来实现类型的参数化。 2. 内存管理:Haskell的容器类型天生支持垃圾回收,无需手动管理内存。而C的泛型需要手动分配和释放内存,并且容易出现内存泄漏和野指针等问题。 3. 表达能力:Haskell的容器类型可以很方便地表示和操作各种数据结构,比如列表、数组、映射等,同时还支持强大的模式匹配和高阶函数。而C的泛型主要用于实现通用算法和数据结构,其表达能力相对较弱。 4. 可靠性和安全性:Haskell的容器类型由于静态类型检查的特性,能够在编译阶段发现大部分类型错误,提高代码的可靠性和安全性。而C的泛型则更容易出现类型相关的错误,并且由于指针操作和类型转换等特性,可能导致运行时错误和安全漏洞。 综上所述,Haskell的容器类型和C的泛型在语言特性、内存管理、表达能力、可靠性和安全性等方面存在显著差异。在使用这两种特性时,需要根据具体的需求和编程语言的特性进行选择。 ### 回答3: Haskell的容器类型与C语言的泛型有几个主要区别。 首先,Haskell的容器类型是静态类型的,而C语言的泛型是动态类型的。在Haskell中,每个容器类型都有一个特定的静态类型,比如List、Set或Map等。这意味着我们在编译时就可以确定容器中元素的类型,并在类型系统的帮助下进行类型安全的操作。而在C语言中,泛型是通过使用void指针来实现的,需要在运行时进行类型检查和转换,这可能会导致类型错误和性能下降。 其次,Haskell的容器类型支持高阶函数,而C语言的泛型不直接支持。在Haskell中,我们可以使用高阶函数来对容器进行各种操作,比如映射、过滤、折叠等。这使得代码更加简洁和可读,并且可以利用很多高级抽象来解决常见的问题。而在C语言中,泛型的实现相对较为低级,我们需要手动编写特定类型的函数来处理容器中的元素。 最后,Haskell的容器类型具有更强大的类型推导能力。在Haskell中,编译器可以自动推导出容器中元素的类型,从而减少了显式类型标注的需求。这使得代码更加简洁、易于编写和理解。而在C语言中,泛型的使用需要显式地指定类型,增加了代码的冗余。 综上所述,Haskell的容器类型与C语言的泛型相比,具有静态类型、支持高阶函数和更强大的类型推导等优势。这些特性使得Haskell为一个强大的函数式编程语言,并提供了便捷而高效的容器操作方式。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值