C++ primer 第十章-泛型算法(Generic Algorithms)

Hi!这里是山幺幺的c++ primer系列。写这个系列的初衷是,虽然在学校学习了c++,但总觉得对这门语言了解不深,因此我想来啃啃著名的c++ primer,并在这里同步记录我的学习笔记。由于我啃的是英文版,所以笔记是中英文夹杂的那种。另外由于已有一定的编程基础,所以这个系列不会含有太基础的知识,想入门的朋友最好自己啃书嘻嘻~

概述


  • 定义:泛型算法中的“算法”是指implement common classical algorithms such as sorting and searching,“泛型”是指operate on elements of differing type and across multiple container types—not only library types such as vector or list, but also the built-in array type—and over other kinds of sequences as well
  • 大部分定义在头文件algorithm中,有的定义在头文件numeric中
  • library algorithms operate on iterators, not containers,所以Algorithms Never Execute Container Operations,所以不会改变容器大小、不会增加/删除元素,而只可能改变容器内元素的值、move elements around within the container等
    PS:若algorithm operates on insert iterators,the iterator may have the effect of adding elements to the container. The algorithm itself, however, never does so.
  • 书上的附录A记录了所有的泛型算法

Read-Only Algorithms


栗子

  • find
  • count
  • accumulate
  • equal

find

  • 格式:find(iter1, iter2, val)
  • find的过程:遍历[iter1, iter2)中的每个元素,与val比较;find must stop when it has reached the end of the sequence
  • 返回值
    • 若找到了,返回an iterator to the first element that is equal to that value
    • 若没找到,返回find函数的第二个参数iter2
  • 适用于vector, list, array等
    auto result = find(vec.cbegin(), vec.cend(), val);
    auto result = find(lst.cbegin(), lst.cend(), val);
    int ia[] = {27, 210, 12, 47, 109, 83};
    int* result = find(begin(ia), end(ia), val);
    auto result = find(ia + 1, ia + 4, val);
    

accumulate

  • 栗子
    // sum the elements in vec starting the summation with the value 0
    int sum = accumulate(vec.cbegin(), vec.cend(), 0);
    
  • The type of the third argument to accumulate determines which addition
    operator is used and is the type that accumulate returns
    string sum = accumulate(v.cbegin(), v.cend(), string(""));
    // error: no + on const char*
    string sum = accumulate(v.cbegin(), v.cend(), "");
    
  • the elements in the sequence must match or be convertible to the type of the third argument

equal

  • 返回值:It returns true if the corresponding elements are equal, false otherwise
  • 栗子
    // roster2 should have at least as many elements as roster1
    equal(roster1.cbegin(), roster1.cend(), roster2.cbegin());
    // 前两个参数denote the range of elements in the first sequence
    // 第三个参数denote the first element in the second sequence
    
  • 可以用equal来比较elements in containers of different types,且the element types also need not be the same so long as we can use == to compare the element types;所以roster1、roster2可以分别是vector<string>、list<const char*>

PS:Algorithms that take a single iterator denoting a second sequence assume that the second sequence is at least as large at the first.

Algorithms That Write Container Elements


注意

  • 必须保证the sequence into which the algorithm writes is at least as large as the number of elements we ask the algorithm to write,因为泛型算法无法改变容器大小

fill

  • 栗子
    fill(vec.begin(), vec.end(), 0); // reset each element to 0
    // set a subsequence of the container to 10
    fill(vec.begin(), vec.begin() + vec.size()/2, 10);
    

fill_n

  • 栗子
    fill_n(vec.begin(), vec.size(), 0); // reset all the elements of vec to 0
    
  • 注意:一定要保证容器中含有足够的元素,因为fill_n是改变现有元素的值,而不是增加元素
    vector<int> vec; // empty vector
    // disaster: attempts to write to ten (nonexistent) elements in vec
    fill_n(vec.begin(), 10, 0);
    
  • 配合使用back_inserter会更安全
    vector<int> vec; // empty vector
    // ok: back_inserter creates an insert iterator that adds elements to vec
    fill_n(back_inserter(vec), 10, 0); // appends ten elements to vec
    

copy

  • 栗子
    int a1[] = {0,1,2,3,4,5,6,7,8,9};
    int a2[sizeof(a1)/sizeof(*a1)]; // a2 has the same size as a1
    // ret points just past the last element copied into a2
    auto ret = copy(begin(a1), end(a1), a2); // copy a1 into a2
    
    list<int> 1st = {1,2,3,4};
    list<int> lst2, lst3; // empty lists
    // after copy completes, 1st2 contains 4 3 2 1
    copy(1st.cbegin(), lst.cend(), front_inserter(lst2));
    // after copy completes, 1st3 contains 1 2 3 4
    copy(1st.cbegin(), lst.cend(), inserter(lst3, lst3.begin()));
    

replace和replace_copy

  • 栗子
// replace any element with the value 0 with 42
replace(ilst.begin(), ilst.end(), 0, 42);

// use back_inserter to grow destination as needed
replace_copy(ilst.cbegin(), ilst.cend(), back_inserter(ivec), 0, 42);
  • replace后ilst中的0会变为42
  • replace_copy后ilst is unchanged, and ivec contains a copy of ilst with the exception that every element in ilst with the value 0 has the value 42 in ivec

Algorithms That Reorder Container Elements


sort

  • 不稳定排序
sort(words.begin(), words.end());

stable_sort

  • 稳定排序

unique

auto end_unique = unique(words.begin(), words.end());
  • 作用:reorder the sequence so that the unique elements appear in the first part of the sequence
  • 返回:the iterator returned by unique denotes one past the last unique element
  • 重复的元素并没有被删除,只是被overwrite了,容器的大小没有变

Predicates


定义

  • an expression that can be called and that returns a value that can be used as a condition
  • The predicates used by library algorithms are either unary predicates (meaning they have a single parameter) or binary predicates (meaning they have two parameters)

callable object


定义

  • An object or expression is callable if we can apply the call operator (即一对括号) to it. That is, if e is a callable expression, we can write e(args) where args is a comma-separated list of zero or more arguments

栗子

  • 函数
  • function pointers
  • classes that overload the function-call operator
  • lambda expressions

Lambda Expressions


定义及特点

  • A lambda expression represents a callable unit of code. It can be thought of as an unnamed, inline function
  • 可以定义在函数内部
  • 可以作为函数的返回值
  • 定义一个lambda时,编译器generates a new (unnamed) class type corresponds to that lambda
  • pass一个lambda给一个函数时,we are defining both a new type and an object of that type: the argument is an unnamed object of this compiler-generated class type
  • 用lambda初始化一个auto变量时,we are defining an object of the type generated from that lambda
  • By default, the class generated from a lambda contains a data member
    corresponding to the variables captured by the lambda,而且the data members of a lambda are initialized when a lambda object is created

格式

  • [capture list] (parameter list) -> return type { function body }
  • capture list:通常是空的,list of nonstatic local variables defined in the surrounding function
  • The capture list is used for local nonstatic variables only; lambdas can use
    local statics and variables declared outside the function directly
  • parameter list and return type是可以省略的
  • 省略return type则使用inferred return type:若函数体中只有return语句,则与其返回的变量类型相同;若函数体中有除return语句外的其他语句,则return type为void
  • lambda expression的形参不能有默认值

栗子

// define f as a callable object that takes no arguments and returns 42
auto f = [] { return 42; };
cout << f() << endl; // prints 42

capture list的使用方法

  • 操作一览
  • capture by value
    • the value of a captured variable is copied when the lambda is created, not when it is called
      void fcn1()
      {
      size_t v1 = 42; // local variable
      // copies v1 into the callable object named f
      auto f = [v1] { return v1; };
      v1 = 0;
      auto j = f(); // j is 42; f stored a copy of v1 when we created it
      }
      
  • caputure by reference
    void fcn2()
    {
    size_t v1 = 42; // local variable
    // the object f2 contains a reference to v1
    auto f2 = [&v1] { return v1; };
    v1 = 0;
    auto j = f2(); // j is 0; f2 refers to v1; it doesn't store it
    }
    
    • 有时需要使用caputure by reference:因为os是一个iostream,不能copy,只能用引用/指针
      for_each(words.begin(), words.end(), [&os, c](const string &s) { os << s << c; });
      
    • When we capture a variable by reference, we must ensure that the variable exists at the time that the lambda executes
  • Implicit Captures
    • 含义:让编译器推断which variables we use from the code in the lambda’s body
    • 用[&]表示是capture by reference,用[=]表示是capture by value
    • 如果想要capture some variables by value and others by reference, we can mix implicit and explicit captures(但the first item in the capture list must be an & or =,表示默认的capture方式)
      // os implicitly captured by reference; c explicitly captured by value
      for_each(words.begin(), words.end(), [&, c](const string &s) { os << s << c; });
      // os explicitly captured by reference; c implicitly captured by value
      for_each(words.begin(), words.end(), [=, &os](const string &s) { os << s << c; });
      

Mutable Lambdas

  • 定义:本来a lambda may not change the value of a variable that it copies by value,但Mutable Lambdas可以
  • 要求:Mutable Lambdas不能省略parameter list
  • 与capture by reference的区别
void fcn3()
{
size_t v1 = 42; // local variable
// f can change the value of the variables it captures
auto f = [v1] () mutable { return ++v1; };
v1 = 0;
auto j = f(); // j is 43
}

void fcn4()
{
size_t v1 = 42; // local variable
// v1 is a reference to a non const variable
// we can change that variable through the reference inside f2
auto f2 = [&v1] { return ++v1; };
v1 = 0;
auto j = f2(); // j is 1
}

指定return type

// compile error: cannot deduce the return type for the lambda(因为inferred return type是void,而又有返回值)
transform(vi.begin(), vi.end(), vi.begin(), [](int i) { if (i < 0) return -i; else return i; });
// 指定return type
transform(vi.begin(), vi.end(), vi.begin(), [](int i) -> int { if (i < 0) return -i; else return i; });

bind


定义

  • 定义在头文件functional中
  • 可以视为general-purpose function adaptor
  • takes a callable object and generates a new callable that “adapts” the parameter list of the original object

格式

  • auto newCallable = bind(callable, arg_list);
  • 含义:调用newCallable时,newCallable调用callable, passing the arguments in arg_list

栗子

bool check_size(const string &s, string::size_type sz)
{
return s.size() >= sz;
}

// check6 is a callable object that takes one argument of type string
// and calls check_size on its given string and the value 6
auto check6 = bind(check_size, _1, 6);

string s = "hello";
bool b1 = check6(s); // check6(s) calls check_size(s, 6)
  • _1是一个place_holder,只有_1表示check6 takes a single argument且the placeholder appears first in arg_list(有两个则写成bind(check_size, _1, _2, 6))
// g(X, Y)等价于f(a, b, Y, c, X)
auto g = bind(f, a, b, _2, c, _1);

使用ref和cref

  • 默认情况下,the arguments to bind that are not placeholders are copied into the callable object that bind returns,如果它不是copyable或者我们想引用而不是复制它,就需要使用ref或cref
  • 定义在头文件functional中
  • ref的作用:returns an object that contains the given reference and that is itself copyable
  • cref的作用:generates a class that holds a reference to const,且copyable
// error: cannot copy os
for_each(words.begin(), words.end(), bind(print, os, _1, ' '));
// 使用ref
for_each(words.begin(), words.end(), bind(print, ref(os), _1, ' '));

Customizing Operations


Passing a Function to an Algorithm(以sort为例)

  • 传predicate:让sort用given predicate in place of < to compare elements
  • 注意:predicate只能take 1或2个参数
  • 栗子
bool isShorter(const string &s1, const string &s2)
{
return s1.size() < s2.size();
}
// sort on word length, shortest to longest
sort(words.begin(), words.end(), isShorter);

Passing a Lambda Expression to an Algorithm

  • 栗子
stable_sort(words.begin(), words.end(),
[](const string &a, const string &b)
{ return a.size() < b.size();});

// get an iterator to the first element whose size() is >= sz
auto wc = find_if(words.begin(), words.end(),
[sz](const string &a)
{ return a.size() >= sz; });

// print words of the given size or longer, each one followed by a space
for_each(wc, words.end(),
[](const string &s){cout << s << " ";});

// 使用bind
// find_if (effectively) will call check_size on each string in the [words.begin(), words.end()) and compare the size of that string to sz
bool check_size(const string &s, string::size_type sz)
{
return s.size() >= sz;
}
using std::placeholders::_1; // 或者直接using namespace std::placeholders;
auto wc = find_if(words.begin(), words.end(), bind(check_size, _1, sz));

// sort on word length, shortest to longest
sort(words.begin(), words.end(), isShorter);
// sort on word length, longest to shortest
sort(words.begin(), words.end(), bind(isShorter, _2, _1));

特殊的迭代器


insert iterator

  • 定义:bound to a container and can be used to insert elements into the container
  • 操作一览
  • 与普通迭代器的区别
    • 普通迭代器:when we assign to a container element through 普通iterator, we assign to the element that iterator denotes(即赋值的前提是iterator指向的元素原本就存在)
    • insert iterator:when we assign through an insert iterator, a new element equal to the right-hand value is added to the container
  • back_inserter
    • 定义在头文件iterator中的函数
    • takes a reference to a container and returns 一个使用push_back的 insert iterator bound to that container
    • 只能用于支持push_back的容器
    • 栗子
    vector<int> vec; // empty vector
    auto it = back_inserter(vec); // assigning through it adds elements to vec
    *it = 42; // vec now has one element with value 42
    
  • front_inserter
    • 与back_inserter类似,但它使用push_front
    • 只能用于支持push_front的容器
  • inserter
    • 与back_inserter类似,但它使用insert
    • 只能用于支持insert的容器
    • 栗子
    * it = val;
    // 等价于
    it = c.insert(it, val); // it points to the newly added element
    ++it; // increment it so that it denotes the same element as before
    

stream iterators

  • 定义:bound to input or output streams and can be used to iterate through the associated IO stream

  • 虽然iostream types并不是容器,但也有iostream iterators

  • 可能有delay:bind an istream_iterator to a stream时,并不一定会立刻读那个stream;只能保证在dereference the iterator for the first time时,stream已被读

  • Operations on istream_iterators

    • 可用于any type that has an input operator (>>)

    • 操作一览

    • 栗子

    ifstream in("afile");
    istream_iterator<string> str_it(in); // reads strings from "afile"
    
    istream_iterator<int> in_iter(cin); // read ints from cin
    istream_iterator<int> eof; // istream ''end'' iterator (hits end-of -file or encounters an IO error)
    while (in_iter != eof) // while there's valid input to read
      vec.push_back(*in_iter++);
    
    istream_iterator<int> in_iter(cin), eof; // read ints from cin
    vector<int> vec(in_iter, eof); // construct vec from an iterator range
    
    istream_iterator<int> in(cin), eof;
    cout << accumulate(in, eof, 0) << endl;
    
  • Operations on ostream_iterators

    • There is no empty or off-the-end ostream_iterator
    • 可用于any type that has an output operator (<<)
    • 操作一览
    • 栗子:以下三段等价;每个输出的元素之间由逗号隔开;++和*不会对ostream_iterator进行任何操作
    ostream_iterator<int> out_iter(cout, ",");
    for (auto e : vec)
      *out_iter++ = e; // the assignment writes this element to cout
    cout << endl;
    
    for (auto e : vec)
      out_iter = e; // the assignment writes this element to cout
    cout << endl;
    
    copy(vec.begin(), vec.end(), out_iter);
    cout << endl;
    

reverse iterators

  • 定义:move backward, rather than forward(forward_list没有reverse iterators)
  • 除forward_list外的容器都支持reverse iterators
  • 不能create a reverse iterator from a stream iterator
  • 栗子
// sorts in reverse: puts the smallest element at the end of vec
sort(vec.rbegin(), vec.rend());

// find the last element in a comma-separated list
auto rcomma = find(line.crbegin(), line.crend(), ',');
// 必须要变回正向迭代器才能按正序打出来
cout << string(rcomma.base(), line.cend()) << endl;
  • 各迭代器之间的关系

move iterators

  • 定义:move rather than copy their elements;the dereference operator of a move iterator yields an rvalue reference
  • 是c++ 11新特性
  • 可以用make_move_iterator函数把普通迭代器转换为move迭代器
void StrVec::reallocate() {
  // allocate space for twice as many elements as the current size
  auto newcapacity = size() ? 2 * size() : 1;
  auto first = alloc.allocate(newcapacity);
  // move the elements
  auto last = uninitialized_copy(make_move_iterator(begin()), make_move_iterator(end()), first);
  free(); // free the old space
  elements = first; // update the pointers
  first_free = last;
  cap = elements + newcapacity;
}

泛型算法的结构


泛型算法需要的五种迭代器

  • input iterators
    • 主要功能:read elements in a sequence
    • 支持:++、==、!=、*(只能出现在赋值语句右边)、->(只能出现在赋值语句右边)
    • 栗子:istream_iterator
    • 适用于:single-pass algorithms like find/accumulate
  • output iterators
    • 主要功能:write elements in a sequence
    • 支持:++、*(只能出现在赋值语句左边)
    • 栗子:ostream_iterator
    • 适用于:single-pass algorithms like copy
  • forward iterators
    • 主要功能:read and write a given sequence(单向)
    • 支持:++、==、!=、*、->
    • 适用于:multi-pass algorithms like replace(因为可以使用saved state of a forward iterator)
    • 栗子:forward_list的迭代器
  • bidirectional iterators
    • 主要功能:read and write a sequence forward or backward
    • 支持:--、++、==、!=、*、->
    • 适用于:multi-pass algorithms like reverse
    • 栗子:除forward_list外的library container的迭代器
  • random-access iterators
    • 主要功能:provide constant-time access to any position in the sequence
    • 支持:--、++、==、!=、*、->、+、-(包括 iter1 - iter2 和 iter1 - n)、+=、-=、>、<、>=、<=、iter[n](相当于 *(iter + n))
    • 适用于:multi-pass algorithms like sort
    • 栗子:array, deque, string, and vector的迭代器

算法参数模板

  • 常用格式:
    alg(beg, end, other args);
    alg(beg, end, dest, other args);
    alg(beg, end, beg2, other args);
    alg(beg, end, beg2, end2, other args);
  • dest
    • an iterator that denotes a destination in which the algorithm can write its output
    • algorithms assume that it is safe to write as many elements as needed,所以dest一般是insert iterator或者ostream_iterator
  • beg2 alone or beg2 and end2
    • denote a second input range
    • 只有beg2的:algorithms assume that the range starting at beg2 is at least as large as the one denoted by beg, end

算法命名规则

  • 在名字后加上 _if,把value替换为predicate
find(beg, end, val); // find the first instance of val in the input range
find_if(beg, end, pred); // find the first instance for which pred is true
  • 在名字后加上 _copy,把rearrange元素的算法变为把rearranged元素写入第二个input range的算法
reverse(beg, end); // reverse the elements in the input range
reverse_copy(beg, end, dest);// copy elements in reverse order into dest
  • remove的栗子
// removes the odd elements from v1
remove_if(v1.begin(), v1.end(), [](int i) { return i % 2; });
// copies only the even elements from v1 into v2; v1 is unchanged
remove_copy_if(v1.begin(), v1.end(), back_inserter(v2), [](int i) { return i % 2; });

list和forward_list-Specific的算法


算法们

  • splice算法
  • 其他特殊算法

与general算法的区别

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