广告
返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >C++primer超详细讲解泛型算法
  • 787
分享到

C++primer超详细讲解泛型算法

2024-04-02 19:04:59 787人浏览 薄情痞子
摘要

目录初识泛型算法只读算法写容器算法定制操作lambda表达式lambda捕获和返回再探迭代器插入迭代器iOStream迭代器反向迭代器初识泛型算法 只读算法 只读取输入范围内的函数,

初识泛型算法

只读算法

只读取输入范围内的函数,不改变元素,find,accumula也是如此

(1)accumulate算法为求和算法,前两个参数指出求和元素范围,第三个是和的初值,例:

int sum=accumulate(v.begin(),v.end(),0)

(2)操作两个序列的算法

equal算法,确定两个序列是否保存相同的值,将第一个序列的每个元素和第二个序列中的每个元素进行比较,若相同返回true,否则返回false,接受三个参数,前两个表示第一个序列的元素范围,第三个表示第二个序列的首元素

equal(r1.begin(),r1.end(),r2.begin())

写容器算法

(1)拷贝算法

向另一个目的位置迭代器指向的输出序列中的元素写入数据算法。此算法接受三个迭代器,前两个表示一个舒服范围,第三个表示目的序列的起始位置。copy返回目的迭代器的值。

int a1[] = { 0,1,2,3,4,5 };
int a2[sizeof(a1) / sizeof(*a1)];
auto ret = copy(begin(a1), end(a1), a2);

定制操作

lambda表达式

(1)定义

一个lambda表达式表示一个可调用的代码单元,可理解为未命名的内联函数

lambda表达式形式:

[capture list](parameter list) - > return type{function body}

可以忽略参数列表和返回类型,但必须永远包含捕获列表和函数体

auto f =[] {return 42}

调用: cout<<f()<<endl;

(2)向lambda传递参数

实参被用来初始化lambda的形参,lambda不能有默认参数

例:

//[](const string &a,const string &b){return a.size()<b.size()}
//调用
sort(w.begin(),w.end(),[](const string &a,const string &b){return a.size()<b.size()})

(3)使用捕获列表

一个lambda通过将局部变量包含在其捕获列表中指出将会使用这些变量,捕获列表指引lambda在其内部包含访问局部变量所需的全部信息

例如,找出第一个大于等于给定长度的单词。

函数biggies实现

void biggies(vector<string> &Words,vector<string>::size_type sz)
{
    auto wc=find_if(words.begin(),words.end(),
                    [sz](const string &a){return a.size()>=sz});
}

lambda捕获和返回

(1)值捕获

与传值参数类似,采用值捕获的前提是变量可以拷贝,与参数不同,被捕获的变量的值是在lambda创建时拷贝,而不是调用时拷贝

void fcun()
{
    size_t v1=42;
    auto f=[v1]{return v1;};
    v1=0;
    auto j=f();
}

由于被捕获变量的值是在lambda创建时拷贝,因此随后对其修改不会影响到lambda内对应的值,上述中j的值为42

(2)引用捕获

void fcun()
{
    size_t v1=42;
    auto f=[&v1]{return v1;};
    v1=0;
    auto j=f();   //j为0,f2保存v1的引用,而非拷贝
}

(3)隐式捕获

可以让编译器根据lambda体中的代码推断我们要使用哪些变量,此时应在捕获列表中写一个&或=,&表示采用捕获引用,=表示采用值捕获方式。

wc=find_if(w.begin(),w.end(),[=](const string &s){return s.size>=sz;});

(4)可变lambda

在值拷贝的情况下,lambda不会改变其值,若希望能够改变被捕获的变量的值,则在参数列表首加上mutable关键字。

void func()
{
    size_t v1=42;
    auto f=[vi]() mutable{return ++v1;};
    v1=0;
    auto j=f();
}

一个引用捕获的变量能否可以修改依赖于此引用指向的是一个const还是非const类型

void fc()
{
	const size_t v1 = 42;
	auto f = [&v1]() mutable {return ++v1; };//无法修改v1
	auto j = f();
	cout << j << endl;
}

(5)指定lambda的返回类型

默认情况下,如果一个lambda体中包含return之外的任何语句,则编译器假定此lambda返回void。

例如:将一个序列中负数替换为其绝对值

transfORM(v.begin(),v.end(),v.begin(),[](int i){return i<0?-i:i;})

transform接受三个参数,前两个表示迭代器输入序列,第三个表示迭代器目的位置。

上述中,我们无需指定返回类型,因此可以根据条件运算符的类型推断出来。

若改写成存在if语句,则存在错误

transform(v.begin(),v.end(),v.begin(),[](int i){if(i<0) return -i; else return i;})

编译器发现存在return之外的语句,所以推断其返回类型为void,而他却返回了一个int

正确写法:

transform(v.begin(),v.end(),v.begin(),
                [](int i) ->int {if(i<0) return -i; else return i;})

再探迭代器

标准库头文件iterator中还定义了额外几种迭代器,包括如下:

  • 插入迭代器:绑定到一个容器上,向容器插入元素
  • 流迭代器:绑定到输入输出流上,可以用来遍历所关联的IO流
  • 反向迭代器:这些迭代器向后而不是向前移动,forwar_list不存在此迭代器
  • 移动迭代器:移动元素专用

插入迭代器

例:

list<int> lst = { 1,2,3,4 };
list<int> lst2, lst3;
copy(lst.begin(), lst.end(), front_inserter(lst2));
copy(lst.begin(), lst.end(), inserter(lst3, lst3.begin()));

front_inserter总是插入容器的第一个元素之前,所以lst2中的元素顺序为4,3,2,1,而lst3中的元素顺序为1,2,3,4

iostream迭代器

istream_iterator读取输入流,ostream_iterator向一个输出流写数据。

(1)istream_iterator操作

创建流迭代器时,必须指定迭代器将要读写的对象类型,可以创建空的初始化迭代器,用作尾后值使用的迭代器,一旦关联的流遇到文件尾或者IO错误,迭代器的值就与尾后迭代器相等。

例:读取输入数据保存到vector中

vector<int> v;
istream_iterator<int> it(cin);
istream_iterator<int> e;
while (it != e)
    v.push_back(*it++);   //或 v.push_back(it,e)

(2)使用算法操作流迭代器

例:

istream_iterator<int> it(cin);
istream_iterator<int> e;
cout << accumulate(it, e, 0) << endl;

(3)ostream_iterator操作

创建一个ostream_iterator时,可以提供第二参数,他是一个字符串,在输出每个元素后都会打印此字符串,不允许空的或表示尾后的ostream_iterator

vector<int> v = { 1,2,3,4 };
ostream_iterator<int> out(cout, "@@");
for (auto e : v)
	*out++ = e;  //赋值语句实际上是将元素写到cout,且*和++并不做任何事
cout << endl;

反向迭代器

反向迭代器就是在容器中尾元素向首元素反向移动的迭代器,递增一个反向迭代器会向前移动一个元素,递减一个迭代器会向后移动一个元素

将反向迭代器转换为普通迭代器可使用reverse_iterator中的base成员函数来完成转换

例:输出最后一个单词

string line = { "first,middle,last" };
auto r = find(line.rbegin(), line.rend(), ',');
cout << string(r.base(), line.end()) << endl;

到此这篇关于c++ primer超详细讲解泛型算法的文章就介绍到这了,更多相关C++ primer泛型算法内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: C++primer超详细讲解泛型算法

本文链接: https://www.lsjlt.com/news/153144.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • C++primer超详细讲解泛型算法
    目录初识泛型算法只读算法写容器算法定制操作lambda表达式lambda捕获和返回再探迭代器插入迭代器iostream迭代器反向迭代器初识泛型算法 只读算法 只读取输入范围内的函数,...
    99+
    2022-11-13
  • C++超详细讲解泛型
    目录1.了解泛型编程2.函数模板2.1简单示例2.2多个模板参数2.3模板实例化2.4模板和普通函数同时存在2.5函数模板不支持定义和声明分离3.类模板3.1简单示例3.2成员函数声...
    99+
    2022-11-13
  • C++ primer超详细讲解关联容器
    目录使用关联容器关联容器概述定义关联容器pair类型关联容器操作关联容器迭代器添加元素删除元素map的下标操作访问元素允许重复关键字的容器的名字开中都有包含单词multi,不保持关键...
    99+
    2022-11-13
  • C++primer超详细讲解顺序容器
    目录顺序容器概述容器库概览迭代器容器定义和初始化赋值和swap顺序容器操作向顺序容器添加元素访问元素删除元素特殊的forwa_list单向链表操作改变容器大小vector对象是如何增...
    99+
    2022-11-13
  • Java 泛型超详细入门讲解
    目录1、什么是泛型2、泛型是怎么编译的泛型的编译机制:擦除机制1、什么是泛型 泛型其实就是将类型作为参数传递,泛型允许程序员在编写代码时使用一些以后才指定的类型 ,在实例化该类时将想...
    99+
    2022-11-13
  • C++BoostGraph算法超详细精讲
    Boost.Graph 中的算法类似于标准库中的算法——它们是通用的并且非常灵活。但是,并不总是很清楚应该如何使用它们。 示例 31.8。使用breadth_...
    99+
    2022-11-13
    C++ Boost Graph C++ Boost Graph算法
  • C++BoostAlgorithm算法超详细精讲
    目录一、说明Boost.Algorithm二、示例练习一、说明Boost.Algorithm Boost.Algorithm 请注意,其他 Boost 库提供了许多算法。例如,您会在...
    99+
    2022-11-13
    C++ Boost Algorithm C++ Boost Algorithm算法
  • C++ Boost CircularBuffer算法超详细精讲
    提要 库 Boost.CircularBuffer 提供了一个循环缓冲区,它是一个具有以下两个基本属性的容器: 循环缓冲区的容量是恒定的,由您设置。当您调用成员函数(例如 push_...
    99+
    2022-11-13
    C++ Boost CircularBuffer Boost.CircularBuffer C++ CircularBuffer
  • Java 泛型详解(超详细的java泛型方法解析)
    目录2. 什么是泛型3. 使用泛型的好处4. 泛型的使用4.1 泛型类4.2 泛型方法4.3 泛型接口5. 泛型通配符5.1 通配符基本使用5.2 通配符高级使用6. 总结1. 为什...
    99+
    2022-11-12
  • React diff算法超详细讲解
    目录diff 算法介绍diff 策略tree diffcomponent diffelement diff结合源码看 diff整体流程新内容为 REACT_ELEMENT_TYPE新...
    99+
    2022-11-13
    React diff算法原理 React diff算法源码
  • C++超详细讲解运算符重载
    目录概念赋值运算符重载const成员取地址及const取地址操作符重载概念 C++为了增强代码的可读性引入了运算符重载,运算符重载是具有特殊函数名的函数,也具有其返回值类 型,函数名...
    99+
    2022-11-13
  • C++BoostUtility超详细讲解
    目录一、说明二、Boost.Utility库示例和代码一、说明 Boost.Utility 库是杂项、有用的类和函数的集合,它们太小而无法在独立库中维护。虽然实用程序很小并且可以快速...
    99+
    2022-12-08
    C++ Boost Utility C++ Utility库
  • C++BoostUuid超详细讲解
    目录一、说明二、Boost.Uuid库示例和代码一、说明 Boost.Uuid 为 UUID 提供生成器。 UUID 是不依赖于中央协调实例的通用唯一标识符。例如,没有数据库存储所有...
    99+
    2022-12-08
    C++ Boost Uuid C++ Uuid标识符
  • C语言超详细讲解排序算法上篇
    目录1、直接插入排序2、希尔排序(缩小增量排序)3、直接选择排序4、堆排序进入正式内容之前,我们先了解下初阶常见的排序分类 :我们今天讲前四个! 1、直接插入排序 基本思...
    99+
    2022-11-13
  • C语言超详细讲解排序算法下篇
    目录1、冒泡排序2、快速排序 ( 三种方法 )3、归并排序4、排序算法复杂度及稳定性分析 上期学习完了前四个排序,这期我们来学习剩下的三个排序 1、冒泡排序 &n...
    99+
    2022-11-13
  • Java贪心算法超详细讲解
    目录什么是贪心算法通过场景理解算法问题分析总结什么是贪心算法 在分析和求解某个问题时,在每一步的计算选择上都是最优的或者最好的,通过这种方式期望最终的计算的结果也是最优的。也就是说,...
    99+
    2022-11-13
  • C++超详细讲解强制类型转换
    目录1 C 强制类型转换2 C++ 强制类型转转1 C 强制类型转换 C 方式的强制类型转换的用法如下代码所示: (Type)(Expression) Type:需要转换成的类型Ex...
    99+
    2022-11-13
  • C++超详细讲解强制类型转换的用法
    目录static_castdynamic_castconst_castreinterpret_caststatic_cast static_cast<type-id>(e...
    99+
    2022-11-13
  • C语言超详细讲解递归算法汉诺塔
    目录题目描述画图分析思路总结代码实现总结题目描述 汉诺塔问题起源于一个传说 汉诺塔又被称为河内塔,传说,在世界中心贝拿勒斯(在印度北部)的圣庙里,一块黄铜板上插着三根宝石针。 印度教...
    99+
    2022-11-13
  • C++ Boost Assign超详细讲解
    目录说明Exercise说明 Boost.Assign Boost.Assign 库提供了帮助函数来初始化容器或向容器添加元素。如果需要将许多元素存储在一个容器中,这些函数尤其有用。...
    99+
    2022-12-09
    C++ Boost Assign C++ Assign库
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作