c++学习心得

仅用于记录平时学习的心得,经常更新

迭代器

  1. iterator优先于const_iterator,reverse_iterator,const_reverse_iterator,因为基本上大多数容器函数都是以iterator为参数,不支持其它三种迭代器,iterator转为const_iterator比较方便,但是反过来转换编译器就会报错,即使使用const_cast也无济于事,所以尽量还是优先使用iterator,如果真的得到了一个const_iterator,函数却需要iterator,可以使用advance和distance搭配来"转换".
using IntDeque = deque<int>;
using Iter = IntDeque::iterator;
using ConstIter = IntDeque::const_iterator;

IntDeque d;
ConstIter ci;
Iter i(d.begin());
advance(i, distance<ConstIter>(i, ci));

distance用于取得两个指向同一个容器的迭代器之间的距离
advance用于将一个迭代器移动指定的距离
注意这里distance使用了模板,需要显式指明参数类型,将两个迭代器统一看成了一种迭代器,如果指明类型,以上将是两种迭代器,模板推导出错,编译失败。
至于这种转换的效率如何,取决于迭代器是随机访问迭代器还是其它双向迭代器等。

  1. reverse_iterator转成iterator的一些问题,需要明白转为iterator的目的,用于插入操作还是用于删除操作,用于插入操作比较方便,直接调用base()函数转换再插入即可,这里涉及到转换后位置下标对应的问题,需要画图说明(不会markdown画图),网上搜base()位置的问题即可,如果涉及删除操作需要将迭代器首先递增一次再转换删除.
vector<int> v;
vector<int>::reverse_iterator ri;
v.erase((++ri).base());
  1. istreambuf_iterator vs istream_iterator, 对于逐个字符的输入考虑使用istreambuf_iterator,该迭代器是直接从流的缓冲区中读去下一个字符,速度快,istream_iterator类似于>>操作符,这会忽略空格字符,而且这个迭代器会考虑很多,会检查很多异常情况,如果只是想从输入流中读取一些字符,大材小用而且效率低,如果想要禁止忽略空格可以设置
ifstream file("xxx.txt");
file.unsetf(ios::skipws);
string fileData((istream_iterator<char>(file)), istream_iterator<char>());

而直接使用istreambuf_iterator因为直接从缓冲区里读,不会忽略任何字符

ifstream file("xxx.txt");
string fileData((istreambuf_iterator<char>(file)), istreambuf_iterator<char>());

算法

  1. transform
int func(int x);
vector<int> values;
vector<int> results;
transform(values.begin(), values.end(), back_inserter(results), func);

这段代码是对values的每个元素执行func函数,把结果插入到results尾部
back_inserter适用于支持push_back的容器
front_inserter适用于支持push_front的容器

  1. 排序stl的选择
  • 如果需要对vector, string, deque或数组中元素进行全排序,可以使用sort或者stable_sort.
  • 对vector, string, deque或数组这种支持随机访问迭代器的数据结构,如果只需要找出前n个元素,并且这n个元素是有顺序的,可以使用partial_sort.
  • 对vector, string ,deque或数组这种,如果只需要找出前n个元素,但是不关心这n个元素的顺序,不需要这n个元素是有序的,可以使用nth_element.
  • 如果需要将一个标准序列容器中的元素按照是否满足某个特定条件区分开来,例如找出前20%的元素,可以使用partition或stable_partition
  • 对于list,仍然可以使用partition和stable_partition算法,可以用list::sort替代sort和stable_sort,但是不支持partial_sort和nth_element,这种算法只支持随机访问迭代器。
  1. remove函数,remove只是将要删除的元素移到了最后,并没有真正的删除,真正删除需要再调用erase.
// 一般如下调用
vector<int> v;
v.erase(remove(v.begin(), v.end(), 5), v.end());
cout << v.size();

但是list是个例外,list中的remove是真的删除,list中的unique也类似,也是真的删除元素。

模板型别推导规则

以如下形式举例
template<typename T>
void f(ParamType param);

f(expr);

共分三种情况:

  • ParamType是指针或者引用但不是万能引用
template<typename T>
void f(T& param);

int x = 1; // x类型int
const int cx = x; // cx类型const int
const int &rx = x; // rx类型const int&

f(x); // T的类型是int, param类型是int&
f(cx); // T的类型是const int,param类型是const int&
f(rx); // T的类型是const int,param类型是const int&
//当ParamType是指针或引用时,引用特性在推导过程中是被忽略的。

template<typename T>
void f(const T& param);
int x = 1;
const int cx = x;
const int &rx = x;

f(x); // T的类型是int, param类型是const int&
f(cx); // T的类型是int,param类型是const int&
f(rx); // T的类型是int,param类型是const int&
//当ParamType是指针或引用时,引用特性在推导过程中是被忽略的。
// 同理,由于param已经具有const特性,所以在推导过程中const属性也会被忽略。

template<typename T>
void f(T* param);
int x = 1;
const int *px = &x;
f(&x); // T的类型int, param类型int*
f(px);  // T的类型是const int , param类型const int*
  • ParamType是万能引用类型
template<typename T>
void f(T&& param);
f(expr);
// 如果expr是个左值,则T和paramtype都会推导为左值引用
// 如果expr是个右值,正常推导
int x = 2;
const int cx = x;
const int& rx = x;

f(x); // x 是左值, T的类型为int&, param 为 int&
f(cx); cx : lvalue, T : const int&, param: const int&
f(rx);rx : lvalue, T: const int&, param: const int&
f(2);2: rvalue, T: int, param : int &&
  • ParamType既不是指针也不是引用
template<typename T>
void f(T param); // param 为值传递
f(expr);
// 因为是值传递,所以expr的所有修饰特性都会被忽略,const, 引用,volatile等,都被忽略

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

推荐阅读更多精彩内容