GeekBand C++ Week7 Notes

Week7 Notes

容器Containers

Stack

也是一个线性容器,是一种先进后出FILO的数据结构,所以支持的操作有push pop,只能访问顶层元素。

#include

int main(){

    std::stacks;

}

底层数据结构是deque为默认的底层结构

templateContainer =

deque<Ty> >

class stack{

}

可以看做是对deque的封装,修改了deque的接口。因为stack不允许遍历,所以没有iterator。也可以用list作为stack的底层数据结构。

Queue队列

是一种先进先出的数据结构,FIFO,有两个出口,只能访问最前或最后的元素,不允许遍历有push,pop,front和back

#include

的层结构是以dequeue作为默认的底层结构

没有iterator

也可以用list

前面都是序列式容器,除了list之外都是连续的内存分布

map and multimap

set and multiset

关联式的容器

元素和元素之间会有某种联系

map存储的对象是key/value pair

不允许有重复的key

春初的对象必须是具备可排序性的

templatePr

= less<Kty>

class _Alloc = allocator > >

class map{…}

map的初始化

struct Employee{

    Employee()

    Employee(conststd::wstring& wszName): Name(wszName) {}

    Std::wstringName;

}

struct ReverseId:public std:binaryfunction{

    booloperator()(cons tint& key1, cons int& key2) const{

        return(key1<=key2)?false:true;

    }

}

const int size = 3;

const std::pairitems[size] = {

    std::make_pair(1,Employee(L”Tom”)),

    std::make_pair(2,Employee(L”Jerry”)),

    std::make_pair(3,Employee(L”Alice”)),

};

std::mapmap1(items, items+3);

插入元素

map1.insert(std::make_pair(4, Employee(L”Brown”)));

map1[5] = Employee(L”Fisher”);

删除元素

std::map::iterator it = map1.begin();

map1.erase(it);

operator[]存取元素

Employee& e = map1[2];

e.SetName(L”…”);

Multimap

类似map的容器,但允许key重复

set and multiset

set是一种关联容器,存储的对象本身既是key又是value,不允许有重复的key,存储的对象必须是可排序性

templatePr =

less<Kty>, classAlloc = allocator<Kty> >

class set{

    …

}

默认less定义排序行为,存储对象必须具备operator<行为,可以自定义排序行为,通过仿函数

std::setps1(personArray, personArray+nSize);

插入元素

ps1.insert(Person(L”Bill”, 4));

删除元素

std::set::iterator it = ps1.begin();

std::advance(it, 1);

ps1.erase(it);

关于set的一些算法

set_union把两个set合并在一起

std::setdest;

std::insert_iterator > ii(dest, dest.begin());

std::set_union(ps1.begin(), ps1.end(),ps2.begin(), ps2.end(), ii, PersonIdComparer);

set_union是依照id来排序的,所以person(L“Apple”,0)仍然跑到了最前面。

Set_intersection找出set中相同的元素

Std::setdest;

Std::insert_iterator > ii(dest, dest.begin());

Std::set_intersection(ps1.begin(),ps1.end(), ps3.begin(), ps3.end(0, ii, PersonIdComparer());

Set_difference

包含在(first1, last1)中而不再(first2, last2)中的不同元素

set需要注意的问题

用于排序的成员(真正的key)

除了真正的key其他成员都可以改变

std::set::iterator it = ps1.find(Person(L”Bill”, 4));

if(it != ps1.end())

it->SetName(L”Bill Gates”);

无法通过编译,set的实现方式不允许通过迭代器改变对象成员

std::set::iterator it = ps1.find(Person(L”Bill”, 4));

if(it != ps1.end())

const_cast(*it).SetName(L”BillGates”)

以下方式无法改

Person tempCopy(*it);

tempCopy.SetName(L”Bill Gates”)

STL整体结构仿函数

STL整体结构

仿函数,适配器

STL组件之间的关系,内存分配器allocator迭代器iterators容器containers算法algorithms仿函数functors适配器adapters

仿函数又称为函数对象function object起作用相当于一个函数指针,remove_if(v.begin, v.end(), ContainsString(L”C++”))

要将某种行为作为算法的函数,此处是判断是否存在C++这几个字符,需要将该行为函数指针作为算法的函数。

STL中将这种行为函数指针定义为所谓的仿函数,其实是一个class再以该仿函数产生一个对象作为算法参数。

STL内建仿函数或者用户自定义仿函数

仿函数的类别定义中必须重载函数调用运算子operator()

#include

std::greater g;

std::cout<

std::cout<()(100,300) <

自定义的仿函数必须重载operator()

struct containsstring: publicstd::unary_function

为什么要用仿函数?不用函数指针?

普通函数指针无法满足stl的抽象要求

函数指针无法和STL其他组件交互

仿函数可以作为模板实参用于定义对象的某种默认行为

以某种顺序对元素进行排序的容器,排序规则就是一个模板实参

templatePr =

less<Ky> >

class set{

    …

}

less作为元素的排序行为,如果两个set具有不同的排序规则,那么进行赋值或者==判断会错误

std::set

> set1, set2; //operator<作为排序行为

std::set

std::greater > set3, set4;//iperator >作为排序行为

if(set1 == set2) //

if(set1 == set3) // error

set1 = set2

set1 = set3 //error

仿函数适配器(functor adapters)

目的在于将无法匹配的函数套接成可以匹配的型别

binder1st/binder2nd

mem_fun/mem_fun_ref

template

class binder1st的源码分析

调用find_if的时候模板里的第二个参数是class具体为not_equal_to,传入的参数是迭代器和输入的数的比较,然后返回find_if

binder2nd,mem_fun, mem_fun_ref

实际使用的时候应当使用更简洁的写法。

Bind1st封装了binder1st的调用复杂性

Template

Inline binder1st <_Fn2> bind1st(const_Fn2& _Func, const _Ty& _Left){

    TypenameFn2::first_argument _typeVal(Left);

    Return (binder1st<_Fn2>(_Func,_Val));

}

类似的还有bind2nd适配器,区别在于_Func操作是作用在左边的值还是右边的值上。

Std::vector::iterator it=

Std::find_if(v.begin(), v.end(),std::bind1st(std::less(), 0));

等价于

std::Vector::iterator it=

std::find_if(v.begin(), v.end(),std::bind2nd(std::greater(), 0));

mem_fun/mem_fun_ref

用来适配对象的成员函数

std::vector v;

std::for_each(v.begin(), v.end(),&Person::Print);//error

有全局函数Print

void PrintPerson(const Person* p){…}

std::for_each(v.begin(), v.end(),&PrintPerson);//OK

可以去调用一个对象

对于函数f以及对象obj,在obj上调用f的形式可以有三种

f(obj)

obj.f()

obj->f()

但在for_each中,只接受第一种形式,mem_fun和mem_fun_ref的存在,是为了obj的成员函数f可以被第一种形式调用

std::for_each(v.begin, v.end(), std::mem_fun(&Person::print));

仿函数适配器

如果vector中存放的不是指针是对象用mem_fun_ref

std::string/std::wstring与vector/vector

单线程情况下对字符串操作用std::string/std::wstring

多线程情况下要考虑是否带引用计数reference count

多线程情况下可以考虑用vector

当new初对象并放入容器时,要在销毁容器之前delete那些对象

std::vector v;

for(vector::iterator it =v.begin(); it != v.end(); it++)

delete (*it);

v.clear;

尽量用算法代替手写循环

通过swap为容器缩水

size和capacity的区别

int array[] = {1, 2, 3, 4, 5, 6, 7, 8, 9,10};

std::vector v(array, array+10);

v.reserve(100000);

::_tprintf(TEXT(“vector size:%d\n”),v.size());// size = 10;

v.capacity() //capacity = 100000

std::vector(v).swap(v);

//SIZE = 10

//capacity = 10

std::vector().swap(v);//清楚v并最小化其容量size =0, capacity = 0

再有对象继承的情况下,建立指针的容器而不是对象的容器

STL装入的对象是原始对象的一个拷贝

如果对象很大,需要大量性能开销

由于继承的存在,拷贝回发生slicing,如果以基类对象建立一个容器而插入派生类对象,那么当对此昂通过基类的拷贝构造函数考入容器的时候对象的派生部分会被切割

泛型算法,非变易算法

泛型算法和内存分配器

非变易算法non-mutating algorithms

变易算法mutating algorithms

排序sorting

泛型数值算法generalized numeric algorithms

非变易算法是一系列模板函数,在不改变操作对象的前提下对元素进行处理,如查找,子序列搜索统计匹配

for_each find find_if count count_ifmismatch equal search find_first_of

非变易算法for_each,调用流程之前已经提过

非变易算法find

int elements [] = {1, 2, 3, 23, 423, 587};

std::vector v(elements,elements+6);

std::vector::iterator it = std::find(v.begin(),v.end(), 23);

if(it != v.end()){

//found!

}

非变易算法find_if

int elements[] = {1,2,3,23,423,596};

std::vector v(elements,elements+6);

std::vector::iterator it =std::find_if(v.begin(), v.end(), std::bind2nd(std::greater(), 100));

非变易算法adjacent_find

非变易算法find_first_of

非变易算法count

非变易算法count_if

非变易算法mismatch

查找*i1 != *(_First2 + i1-_First1)

非变易算法search

变易算法,变易算法是指那些改变容器中对象的操作

copy, swap, transform, replace, fill,generate, remove, unique, reserve, rotate, random_shuffle, partition

变易算法copy, copy_n, copy_backward, copy_if,

变易算法swap

变易算法transform

变易算法replace

变易算法fill

变易算法generate

std::vector v(5);

std::generate(v.begin(), v.end(), rand);

变易算法remove

变易算法unique

变易算法reverse

变易算法rotate

template inline

FwdIt rotate(FwdIt _First, _FwdIt _Mid, _FwdIt _Last)

变易算法partition

泛型算法

排序sort,对于排序的元素需要提供operator<

partial_sort

部分排序,有可能从后面排到前面来,所以排序是一种遍历的算法??

Binary_search,搜索前要先排序

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,163评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,301评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,089评论 0 352
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,093评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,110评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,079评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,005评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,840评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,278评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,497评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,394评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,980评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,628评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,649评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,548评论 2 352

推荐阅读更多精彩内容