(七)其他库
c++还提供了其他一些类库,比如头文件<complex>为复数运算提供了类模板complex,包含用于float,long,long double的具体化,这个类库提供了标准的复数运算和能够处理复数的函数;头文件<random>提供了许多新的随机数的功能。
1.vector,valarray和array
vector类是一个容器类和算法系统的一部分,属于STL,支持面向容器的操作,如排序sort(),插入insert(),重新排列random_shuffle(),搜索find(),将数据转移到其他容器swap()等;
valarray类不属于STL,它是面向数值计算的,因此很多数学运算符都经过了重载,可以使用数乘,加法等操作。Valarray类对象可以直接整体进行运算,这样就是对其中的每一个元素进行相应的运算,对于数值计算来说,这是非常方便的。valarray类还提供了sum(),max(),min(),size()等方法,方便数据的处理;除了这些,valarray还有一些其他特性,其中一个就是slice用作下标的类,slice对象通过3个参数来构造,比如slice(1,4,3),表示有4个元素,起始为1,跨距是3,也就是值为1,4,7,10的所有的下标,因此如果vector aa;那么aa[slice(1,4,3)]就表示下标为1,4,7,10的所有的元素。slice用作数组索引,表示的不是一个值,而是一组值。
array是为了替代内置数组而设计的,通过提供更好更安全的接口,让数组更紧凑,效率更高,array类尽管不是STL的一部分(比如不支持push_back操作),但是它支持很多STL方法(包括begin(),end()等),因此可以使用STL算法对array类进行操作。
2.模板initializer_list
模板initializer_list是c++11新加的功能,也就是将初始化列表作为参数来构建对象,使得这种大括号形式的初始化可以应用于各个方面。
初始化列表,要使用initializer_list对象,需要包含头文件initializer_list。initializer_list本身就是一个类模板,{2,3,5.6}就是一个initializer_list对象,可以使用begin,end等来表示起始元素和超尾元素的位置。也可以像使用普通对象那样来使用initializer_list对象。
很多容器类支持列表初始化语法,主要是因为这些容器类有一个使用initializer_list作为构造参数的构造函数,比如vector aa{22,34.2};之所以可行,是因为vector模板类有一个相应的构造函数,上述代码与下面的代码等价vector aa({22,34.2});initializer_list表示的就是大括号包含的内容。列表初始化语法有一个特点,就是可以进行隐式的拓宽转换,比如int型转换为double型,但是不能进行窄化转换,比如不能将double自动转换为int型。除非类要处理不同长度的数据序列,否则没有必要创建一个以初始化列表作为参数的构造函数。
3.总结
有些算法可以表示为容器类算法,但是大部分算法都被表示为通用的,非成员的函数,这是通过将迭代器作为容器和算法之间的接口而实现的。
STL还提供了函数对象(函数符),函数对象是重载了()运算符的类生成的对象。自适应函数符有typedef语句,这些语句表示了函数符的返回类型和参数类型。
STL提供了非常有用的可重用代码源,可以直接使用STL来解决编程问题,也可以将他们作为基本部件,来构建所需的解决方案。