摘要

本文主要借助对C++的标准模板库STL中实现的数据结构的学习和使用来加深对数据结构的理解,即联系数据结构的理论分析和详细的应用实现(STL),本文是系列总结的第二篇。主要针对线性表中的链表
STL std::list进行分析和总结。

引言

因为前段时间对台大的机器学习基石和技法课程进行了学习,发如今详细的实现中经常涉及到各种类型的数据结构,比方线性表、二叉树、图等。在使用这些数据结构时感到有些吃力,主要是对一些主要的数据结构理解的不够,所以趁着暑假假期,近期一段时间总会抽出时间复习一下数据结构,參考的教材是王立柱编著的《C/C++与数据结构》,在详细的应用中採用的是C++标准模板库STL中相应实现的数据结构,主要參考的是MSDN文档。

跟着教材的一步一步推进。如今已经复习完了链表一章节。详细的理论能够參看我的博文:http://blog.csdn.net/lg1259156776/article/details/47018813

本次关注点在list模板类的使用。

正文

回想动态数组类

上一篇总结STL vector动态数组类的时候忘记了对还有一种跟vector很类似的动态数组类deque进行说明。

以下对此进行一下补充。

STL deque类须要包括<deque>和使用std。支持在数组的开头和末尾插入或删除元素,而vector仅仅能在末尾插入或删除,即仅仅有push_back和pop_back。deque同意使用push_front和pop_front在开头插入和删除元素。其余的操作大致类似,不再赘述!

template <
class Type,
class Allocator=allocator<Type>
>
class deque
<span style="font-size:18px;">// deque_push_front.cpp
// compile with: /EHsc
#include <deque>
#include <iostream>
#include <string> int main( )
{
using namespace std;
deque <int> c1; c1.push_front( 1 );
if ( c1.size( ) != 0 )
cout << "First element: " << c1.front( ) << endl; c1.push_front( 2 );
if ( c1.size( ) != 0 )
cout << "New first element: " << c1.front( ) << endl; // move initialize a deque of strings
deque <string> c2;
string str("a"); c2.push_front( move( str ) );
cout << "Moved first element: " << c2.front( ) << endl;
}</span>

总结:在不知道须要存储多少个元素时,务必使用动态数组vector或deque。并牢记vector仅仅能使用push_back在一端扩容,而deque能够使用push_back和push_front在两端扩容。另外。訪问动态数组时,不要跨越其边界。

list 模板类

标准模板库以模板类std::list的方式向程序猿提供了一个双向链表。

双向链表的主要长处是插入和删除元素的速度快,且时间固定,不像顺序表那样须要移动元素。从C++ 11起,还能够使用单向链表std::forward_list,仅仅沿着一个方向遍历。

template <
class Type,
class Allocator=allocator<Type>
>
class list

list的特点

链表是一系列节点,每一个节点除了包括对象或data之外,还包括指向下一个或者上一个节点的指针。list类的STL实现同意在开头、末尾和中间插入元素,且所需的时间固定。

使用时包括<list>和std。

list的基本操作

实例化

list<int> listIntegers; list<float> listFloats;等等。要声明一个指向list中元素的迭代器,能够进行例如以下操作:

list<int> :: const_iterator iElementInSet; 还记得const_iterator吧,在上一篇vector的分析中讲到过。指向一个仅仅读的元素,假设要对容器中的内容进行改动,能够使用iterator来进行。另一些与vector一样的初始化操作,能够參考上一篇博文,兴许关于STL容器进行编程或者解说时。实例化的方式大致一样。这样的模式也将长期出现,兴许博文便不再多提。

在list开头或末尾插入或删除元素

跟deque类相似。採用push_front/pop_front和push_back/pop_back的方法。

在list中间插入或删除元素

list的特点之中的一个。上面讲过。在中间插入或删除元素所需的时间是固定的,使用函数insert()和erase()。

从上面的分析看,基本上全部的容器类(vector。list,deque...)所使用的方法模式都类似,这对于触类旁通的学习非常有帮助。

list元素的反转和排序

list的一个独到之处就是指向元素的迭代器在list的元素又一次排列或插入元素后仍然有效,为实现这样的特点,list提供了成员方法sort和reverse。尽管STL也提供了这两种算法(在算法类中<algorithm>)。且这些算法相同能够使用在list类上。

使用reverse()反转元素的顺序排列

list提供了成员函数reverse,没有參数,功能是反转list中元素的排列顺序:

<span style="font-size:18px;">// list_reverse.cpp
// compile with: /EHsc
#include <list>
#include <iostream> int main( )
{
using namespace std;
list <int> c1;
list <int>::iterator c1_Iter; c1.push_back( 10 );
c1.push_back( 20 );
c1.push_back( 30 ); cout << "c1 =";
for ( c1_Iter = c1.begin( ); c1_Iter != c1.end( ); c1_Iter++ )
cout << " " << *c1_Iter;
cout << endl; c1.reverse( );
cout << "Reversed c1 =";
for ( c1_Iter = c1.begin( ); c1_Iter != c1.end( ); c1_Iter++ )
cout << " " << *c1_Iter;
cout << endl;
}</span>

输出结果:

c1 = 10 20 30
Reversed c1 = 30 20 10

使用sort()对元素进行排序

list成员函数sort有两个版本号。一是没有參数,默认是升序排列。假设想降序排列,能够升序后採用reverse。还有一个版本号是接受一个二元谓词函数作为參数。制定排序的标准。比方以下一个二元谓词函数GeaterThan,指定的排序标准是降序排列

bool GeaterThan(const int& lsh, const int& rsh)

{

return(lsh > rsh);

}

<span style="font-size:18px;">// list_sort.cpp
// compile with: /EHsc
#include <list>
#include <iostream> int main( )
{
using namespace std;
list <int> c1;
list <int>::iterator c1_Iter; c1.push_back( 20 );
c1.push_back( 10 );
c1.push_back( 30 ); cout << "Before sorting: c1 =";
for ( c1_Iter = c1.begin( ); c1_Iter != c1.end( ); c1_Iter++ )
cout << " " << *c1_Iter;
cout << endl; c1.sort( );
cout << "After sorting c1 =";
for ( c1_Iter = c1.begin( ); c1_Iter != c1.end( ); c1_Iter++ )
cout << " " << *c1_Iter;
cout << endl; c1.sort( greater<int>( ) );
cout << "After sorting with 'greater than' operation, c1 =";
for ( c1_Iter = c1.begin( ); c1_Iter != c1.end( ); c1_Iter++ )
cout << " " << *c1_Iter;
cout << endl;
}</span>

输出为:

Before sorting: c1 = 20 10 30
After sorting c1 = 10 20 30
After sorting with 'greater than' operation, c1 = 30 20 10

对包括对象的list进行排序及删除当中的元素

在实际应用非常少使用STL容器来存储整数,而是存储用户自定义的类型,比方类或结构等。那么这样的排序怎样做呢?

比方list中存储的是电话簿,当中每一个元素都是一个对象,包括名称、地址等内容,怎样确保能依照名称对其进行排序呢?

答案是採取下面两种方式之中的一个:

<1> 在list包括相应所属的类中,实现运算符<

<2> 提供一个排序二元谓词(一个函数。接受两个输入值,并返回一个bool值。指出第一个值是否比第二个值小)

在实际的sort调用中。首先推断有没有输入二元谓词,假设没有sort函数检查相应的list的对象元素中是否实现了运算符<。假设实现了则依照该运算符指定的排序标准进行排序。假设输入了二元谓词函数。则依照其标准进行排序。

在用remove进行删除的时候也是一样,仅仅须要提供给元素对象中的某一个变量。就能够直接删除全部的对象中该变量值相等的元素。

即remove函数须要提供一个匹配标准才行,在对象类中实现==运算符。就是为remove时提供的删除标准。比方电话簿中的名字,在类中实现一个例如以下所看到的的重载运算符==

bool operator == (const ContactItem& itemToCompare) const

{

return(itemToCompare.strContactsName == this->strContactsName);

}

forward_list模板类

从c++ 11之后,提供了forward_list来支持单向链表,包括头文件<forward_lsit>。用法与list非常类似,就好像vector与deque的关系。forward_list仅仅能沿着一个方向移动迭代器,且插入元素的时候仅仅能使用函数push_front(),而不能使用push_back。当然是用insert是能够在指定位置插入元素的。

总结

假设须要频繁的插入和删除元素,尤其是在中间插入或删除,应当使用std::list,而不是使用std::vector。由于在这样的情况下vector须要调整内部缓冲区大小。以支持数组语法。还需运行开销高昂的复制操作。而list仅仅需建立或断开链接。

对于使用list等STL容器存储其对象的类,别忘了在当中实现运算符<和==,以提供默认排序的标准和删除谓词。

当无需频繁插入和删除元素时。请不要使用list。使用vector和deque的速度要更快。假设不想依据默认标准进行删除或排序,别忘了给sort和remove提供一个谓词函数。

*************************************************************************************************************************************

2015-7-23

C++的标准模板库STL中实现的数据结构之链表std::list的分析与使用的更多相关文章

  1. C++的标准模板库STL中实现的数据结构之顺序表vector的分析与使用

    摘要 本文主要借助对C++的标准模板库STL中实现的数据结构的学习和使用来加深对数据结构的理解.即联系数据结构的理论分析和详细的应用实现(STL),本文是系列总结的第一篇,主要针对线性表中的顺序表(动 ...

  2. 实验8 标准模板库STL

    一.实验目的与要求: 了解标准模板库STL中的容器.迭代器.函数对象和算法等基本概念. 掌握STL,并能应用STL解决实际问题. 二.实验过程: 完成实验8标准模板库STL中练习题,见:http:// ...

  3. 标准模板库(STL)学习探究之vector容器

    标准模板库(STL)学习探究之vector容器  C++ Vectors vector是C++标准模板库中的部分内容,它是一个多功能的,能够操作多种数据结构和算法的模板类和函数库.vector之所以被 ...

  4. C++ 标准模板库(STL)

    C++ 标准模板库(STL)C++ STL (Standard Template Library标准模板库) 是通用类模板和算法的集合,它提供给程序员一些标准的数据结构的实现如 queues(队列), ...

  5. STL学习系列之一——标准模板库STL介绍

    库是一系列程序组件的集合,他们可以在不同的程序中重复使用.C++语言按照传统的习惯,提供了由各种各样的函数组成的库,用于完成诸如输入/输出.数学计算等功能. 1. STL介绍 标准模板库STL是当今每 ...

  6. 标准模板库--STL

    标准模板库STL 1.泛型程序设计 C++ 语言的核心优势之一就是便于软件的重用 C++中有两个方面体现重用: 1.面向对象的思想:继承和多态,标准类库 2.泛型程序设计(generic progra ...

  7. C++ 标准模板库STL 队列 queue 使用方法与应用介绍

    C++ 标准模板库STL 队列 queue 使用方法与应用介绍 queue queue模板类的定义在<queue>头文件中. 与stack模板类很相似,queue模板类也需要两个模板参数, ...

  8. 标准模板库(STL)学习探究之stack

    标准模板库(STL)学习探究之stack queue priority_queue list map/multimap dequeue string

  9. cb22a_c++_标准模板库_STL_map_multimap红黑树(数据结构)关联容器

    cb22a_c++_标准模板库_STL_map_multimap红黑树(数据结构)关联容器map(映射,key不能重复,一对一对的,value_type(1, "one")),mu ...

随机推荐

  1. MFC 与Excel文件的交互操作

    假日快要结束了.带着沉重的心情写下之前关于MFC与Excel文件交互的总结. 因为VS的版本号不同可能在操作上有些差异.所以在此指明下本篇文章的project环境为VS2013,也建议大家用最新的. ...

  2. UVA 294 - Divisors 因子个数

    Mathematicians love all sorts of odd properties of numbers. For instance, they consider 945 to be an ...

  3. tp中使用事务

    是什么 事务是为了防止,多个操作,其中有失败,数据有部分被执行成功的时候使用的. 比如,银行,用户转账.张三钱扣了,结果李四钱还增加! 这个时候需要使用事务,确保张三钱扣了,李四的钱也增加,才真正的成 ...

  4. php5.5安装笔记

    这次没想到本来很简单的php编译,没想到遇到那么多问题.再此记录一下. 1.php5.5编译安装主要有一个难点,就是GD库的问题,因为php5.5的GD库必须是2.1以上的版本哦 原来都是用的gd2. ...

  5. 4.matlab基础

    1 函数句柄 clear all; f1=@cos t=:pi/:pi f1(t) f2=@complex f2(,) clear all %函数句柄 f1=@char %函数句柄转换为字符串 s1= ...

  6. .NET MVC Dropzone 上传图片

    在nuget控制台输入:Install-Package dropzone @{ Layout = null; } <!DOCTYPE html> <html> <head ...

  7. Edge 通过代理无法打开网页,解决方案

    netsh winhttp import proxy source=ie

  8. (转载)Android项目实战(三十二):圆角对话框Dialog

    Android项目实战(三十二):圆角对话框Dialog   前言: 项目中多处用到对话框,用系统对话框太难看,就自己写一个自定义对话框. 对话框包括:1.圆角 2.app图标 , 提示文本,关闭对话 ...

  9. 列表查询组件代码, 简化拼接条件SQL语句的麻烦

    列表查询组件代码, 简化拼接条件SQL语句的麻烦 多条件查询

  10. SQL Server-聚焦移除Bookmark Lookup、RID Lookup、Key Lookup提高SQL查询性能

      前言 前面几节都是讲的基础内容,本节我们讲讲索引性能优化,当对大数据进行处理时首先想到的就是索引,一旦遇到这样的问题则手忙脚乱,各种查资料,为何平常不扎实基本功呢,我们由浅入深,简短的内容,深入的 ...