c++泛型算法(一)

简介: c++泛型算法(一)

C++ 泛型算法概述

在 C++ 中,标准模板库(STL)提供了一套强大的泛型算法,这些算法设计用来处理 STL 容器中的数据。泛型算法的“泛型”一词指的是这些算法能够独立于它们所操作对象的具体类型。这些算法大多定义在  和  头文件中。

泛型算法的特点

与容器类型无关:泛型算法可以作用于任何提供迭代器接口的容器,如 std::vector、std::list、std::set 等。

操作通过迭代器进行:算法通常通过迭代器来访问和处理容器中的元素,这使得算法与容器的具体类型解耦。

高度可重用和可定制:通过模板参数和函数对象(比如谓词和比较函数),算法可以被定制以适应不同的处理需求。

常用的泛型算法


  1. 排序和搜索:
  • sort(begin, end):对元素进行排序。
  • binary_search(begin, end, value):在已排序的序列中进行二分查找。

修改序列操作:


copy(begin, end, dest):复制元素到另一个位置。

fill(begin, end, value):用特定值填充序列。

replace(begin, end, old_val, new_val):替换序列中的特定值。

unique(begin, end):移除相邻的重复元素。

分区操作:


partition(begin, end, predicate):根据谓词对序列进行分区。

数值操作(在  中定义):


accumulate(begin, end, init):计算序列中元素的累积总和。

inner_product(begin1, end1, begin2, init):计算两个序列的内积。

查询操作:


count(begin, end, value):计算序列中特定值出现的次数。

find(begin, end, value):在序列中查找特定值。

示例

#include <algorithm>
#include <vector>
#include <iostream>

int main() {
    std::vector<int> vec = {4, 1, 3, 5, 2};

    // 排序
    std::sort(vec.begin(), vec.end());

    // 查找
    bool found = std::binary_search(vec.begin(), vec.end(), 3);

    // 输出
    std::cout << "Sorted vector: ";
    for (int val : vec) {
        std::cout << val << " ";
    }
    std::cout << "\n3 found: " << found << std::endl;

    return 0;
}


输出:

Sorted vector: 1 2 3 4 5 
3 found: 1

向算法传递函数

在 C++ 的标准模板库(STL)中,许多泛型算法允许你传递函数或函数对象,以自定义算法的行为。这种机制增加了算法的灵活性和可重用性,允许你为算法指定特定的操作或自定义的比较、条件判断逻辑。

函数传递的方式

  1. 函数指针:传统的方法是使用指向函数的指针。
  2. 函数对象(Functors):对象的类重载了 operator(),可以像函数一样被调用。
  3. Lambda 表达式:C++11 引入的特性,允许你以匿名函数的方式直接在算法调用处定义函数逻辑。

示例

1. 使用函数指针

bool is_odd(int n) {
    return n % 2 != 0;
}

// 在算法中使用函数指针
std::vector<int> vec = {1, 2, 3, 4, 5};
auto it = std::find_if(vec.begin(), vec.end(), is_odd);

使用函数对象

struct IsOdd {
    bool operator()(int n) const {
        return n % 2 != 0;
    }
};

// 在算法中使用函数对象
IsOdd is_odd_obj;
auto it = std::find_if(vec.begin(), vec.end(), is_odd_obj);

3. 使用 Lambda 表达式

// 在算法中直接使用 Lambda 表达式
auto it = std::find_if(vec.begin(), vec.end(), [](int n) { return n % 2 != 0; });

在所有这三个示例中,std::find_if 算法被用来查找 vec 中的第一个奇数。每个示例都展示了向算法传递函数的不同方式。


为什么使用函数传递

  • 灵活性:你可以在不修改算法本身的情况下,改变算法的行为。
  • 代码重用:可以在不同的上下文中重用相同的算法,应用不同的操作。
  • 表达力:特别是使用 Lambda 表达式,你可以在算法调用的地方直接定义操作逻辑,使代码更加紧凑和易读。

Lambda 表达式

C++ 中的 Lambda 表达式是 C++11 引入的功能,它允许定义匿名函数。Lambda 表达式在 C++ 中非常有用,尤其是在需要简短且一次性的函数对象时,例如作为参数传递给 STL 算法。以下是 Lambda 表达式的主要知识点:

1. 基本语法

Lambda 表达式的基本语法如下:

[ capture_clause ] ( parameters ) -> return_type { body }


捕获子句(capture_clause):定义了 Lambda 表达式从封闭作用域捕获变量的方式。

参数列表(parameters):类似于常规函数的参数列表。

返回类型(return_type):Lambda 表达式的返回类型。在某些情况下可以省略,让编译器自动推导。

函数体(body):包含 Lambda 表达式的代码。

2. 捕获列表

捕获列表定义了 Lambda 表达式可以从封闭作用域中捕获哪些变量,以及如何捕获(值捕获或引用捕获)。

示例

  • 值捕获:[x] 捕获变量 x 的副本。
  • 引用捕获:[&x] 通过引用捕获变量 x。
  • 隐式捕获:[=] 捕获所有外部变量的副本,[&] 捕获所有外部变量的引用。

3. 参数和返回类型

  • 参数用于传递值给 Lambda 表达式。
  • 返回类型可以显式指定,也可以让编译器自动推导。

示例

  • 带参数的 Lambda:[](int x, int y) { return x + y; }
  • 显式指定返回类型:[](int x, int y) -> int { return x + y; }


4. 使用场景

  • 作为参数传递给 STL 算法。
  • 用于定义短小的函数操作,避免定义单独的函数或函数对象。
  • 在需要临时函数对象的场合使用。


5. Lambda 表达式与函数指针

Lambda 表达式可以转换为函数指针,前提是它不捕获任何外部变量。

示例

  • void (*func)(int) = [](int x) { std::cout << x; };

6. 可变 Lambda

使用关键字 mutable 允许在值捕获模式下修改捕获的变量。

示例

  • [x]() mutable { x = 42; }

7. 泛型 Lambda

C++14 引入的特性,使用自动类型推导的参数。

示例

  • auto lambda = [](auto x, auto y) { return x + y; };

C++ 参数绑定

C++ 参数绑定是一种将函数或函数对象的部分参数预先设定(或“绑定”)的技术。这在 C++11 中通过 std::bind 函数模板实现,它位于  头文件中。参数绑定通常用于将已有函数或函数对象的参数接口适配到特定场景的需求。

基本概念

std::bind

std::bind 接受一个可调用对象和一系列参数,返回一个新的可调用对象。新的可调用对象可以用更少的参数调用,因为一些参数已经被绑定。


占位符

std::placeholders 命名空间中的 _1、_2、_3 等是占位符,用于 std::bind 中表示未绑定的参数。

示例

1. 绑定普通函数

#include <functional>
#include <iostream>

void print(int a, int b, int c) {
    std::cout << a << ", " << b << ", " << c << std::endl;
}

int main() {
    auto bindPrint = std::bind(print, std::placeholders::_1, 2, std::placeholders::_2);
    bindPrint(1, 3); // 相当于 print(1, 2, 3);
    return 0;
}

2. 绑定成员函数

class MyClass {
public:
    void memberFunc(int x) {
        std::cout << "Value: " << x << std::endl;
    }
};

int main() {
    MyClass obj;
    auto bindMemberFunc = std::bind(&MyClass::memberFunc, &obj, std::placeholders::_1);
    bindMemberFunc(10); // 相当于 obj.memberFunc(10);
    return 0;
}


3. 绑定函数对象

struct Adder {
    int operator()(int a, int b) {
        return a + b;
    }
};

int main() {
    Adder adder;
    auto bindAdder = std::bind(adder, 5, std::placeholders::_1);
    std::cout << bindAdder(3); // 相当于 adder(5, 3);
    return 0;
}

注意事项

参数拷贝:std::bind 在绑定时会对其参数进行拷贝,除非使用 std::ref 或 std::cref 显式指定引用传递。

可读性:虽然 std::bind 提供了强大的功能,但过度使用可能会降低代码的可读性。在 C++11 及更高版本中,Lambda 表达式通常是更简洁和直观的选择。

再探迭代器

在 C++ 中,除了标准容器自带的迭代器之外,标准库还在  头文件中定义了四种特殊的迭代器。这些迭代器提供了不同于普通迭代器的特殊功能,使得操作容器和 IO 流更加灵活。

插入迭代器

插入迭代器绑定到容器上,使得可以通过迭代器向容器插入元素,而不是修改容器中已有的元素。


类型

back_inserter:创建一个使用 push_back 的迭代器。

front_inserter:创建一个使用 push_front 的迭代器(仅对支持 push_front 的容器有效)。

inserter:创建一个使用 insert 的迭代器,可以在指定位置之前插入元素。

示例

std::vector<int> vec;
std::copy(vec.begin(), vec.end(), std::back_inserter(vec)); // 将 vec 的元素复制到其自身末尾

流迭代器

流迭代器绑定到输入或输出流上,可以用来遍历所关联的 IO 流。

类型


  • istream_iterator:从 std::istream 对象读取数据。
  • ostream_iterator:向 std::ostream 对象写入数据。

示例

std::istream_iterator<int> in_iter(std::cin), eof;
std::vector<int> vec(in_iter, eof); // 从标准输入读取数据填充到 vec

std::ostream_iterator<int> out_iter(std::cout, " ");
std::copy(vec.begin(), vec.end(), out_iter); // 将 vec 的内容输出到标准输出

反向迭代器

反向迭代器使迭代器在容器中向后移动,除了 std::forward_list 外,所有标准库容器都支持反向迭代器。

示例

std::vector<int> v = {1, 2, 3};
for (auto rit = v.rbegin(); rit != v.rend(); ++rit) {
    std::cout << *rit << " "; // 输出:3 2 1
}

移动迭代器

移动迭代器允许在算法中移动而非拷贝元素,这对于管理资源密集型对象的容器特别有用。

示例

std::vector<std::unique_ptr<int>> vec;
// 使用移动迭代器以避免拷贝 unique_ptr
std::vector<std::unique_ptr<int>> vec2(std::make_move_iterator(vec.begin()), 
                                       std::make_move_iterator(vec.end()));

泛型算法结构

C++ 标准模板库(STL)的泛型算法结构是一种高度模块化和可复用的设计,它使得这些算法能够独立于它们所处理的数据类型。泛型算法通常是通过模板实现的,这意味着算法可以用于不同类型的容器和迭代器。以下是泛型算法结构的关键特点:

5 类迭代器

在 STL 中,迭代器被分为五种类别,每种类别定义了迭代器支持的操作集合。这些类别是:

  1. 输入迭代器:仅支持读操作,单向移动。
  2. 输出迭代器:仅支持写操作,单向移动。
  3. 前向迭代器:支持读写操作,单向移动。
  4. 双向迭代器:支持读写操作,可前后双向移动。
  5. 随机访问迭代器:支持读写操作,可在序列中随机访问。

这种分类方式使得泛型算法可以根据所需的迭代器类型来适配于不同的容器类型。

算法形参模式

泛型算法通常接受一对迭代器(表示一个范围)作为参数,这种设计使得算法可以在任何容器类型上操作,只要容器提供了适当类型的迭代器。例如,许多算法都接受形如 begin 和 end 的迭代器参数,用于指定算法作用的元素范围。


算法命名规范

为了表明算法的特定版本或特性,STL 算法通常遵循一定的命名规范。例如:


  • 算法名称后缀 _if 表示算法接受一个谓词(如 find_if)。
  • 算法名称后缀 _copy 表示算法会生成一个拷贝(如 copy_if)。
  • 算法名称后缀 _n 表示算法作用于特定数量的元素(如 copy_n)。


特点与优势


  • 类型独立性:泛型算法可以作用于任何数据类型。
  • 灵活性:算法可以与不同的容器和迭代器类型配合使用。
  • 可重用性:相同的算法可以用于多种数据结构和操作。
  • 效率:泛型算法经过高度优化,能提供良好的性能。


c++泛型算法(二)https://developer.aliyun.com/article/1437192?spm=a2c6h.13262185.profile.35.5bba685cuSQkDD


目录
相关文章
|
1天前
|
存储 算法 测试技术
【C++数据结构——树】二叉树的遍历算法(头歌教学实验平台习题) 【合集】
本任务旨在实现二叉树的遍历,包括先序、中序、后序和层次遍历。首先介绍了二叉树的基本概念与结构定义,并通过C++代码示例展示了如何定义二叉树节点及构建二叉树。接着详细讲解了四种遍历方法的递归实现逻辑,以及层次遍历中队列的应用。最后提供了测试用例和预期输出,确保代码正确性。通过这些内容,帮助读者理解并掌握二叉树遍历的核心思想与实现技巧。
14 2
|
9天前
|
存储 算法 安全
基于红黑树的局域网上网行为控制C++ 算法解析
在当今网络环境中,局域网上网行为控制对企业和学校至关重要。本文探讨了一种基于红黑树数据结构的高效算法,用于管理用户的上网行为,如IP地址、上网时长、访问网站类别和流量使用情况。通过红黑树的自平衡特性,确保了高效的查找、插入和删除操作。文中提供了C++代码示例,展示了如何实现该算法,并强调其在网络管理中的应用价值。
|
7天前
|
存储 算法 安全
基于哈希表的文件共享平台 C++ 算法实现与分析
在数字化时代,文件共享平台不可或缺。本文探讨哈希表在文件共享中的应用,包括原理、优势及C++实现。哈希表通过键值对快速访问文件元数据(如文件名、大小、位置等),查找时间复杂度为O(1),显著提升查找速度和用户体验。代码示例展示了文件上传和搜索功能,实际应用中需解决哈希冲突、动态扩容和线程安全等问题,以优化性能。
|
14天前
|
算法 安全 C++
用 C++ 算法控制员工上网的软件,关键逻辑是啥?来深度解读下
在企业信息化管理中,控制员工上网的软件成为保障网络秩序与提升办公效率的关键工具。该软件基于C++语言,融合红黑树、令牌桶和滑动窗口等算法,实现网址精准过滤、流量均衡分配及异常连接监测。通过高效的数据结构与算法设计,确保企业网络资源优化配置与安全防护升级,同时尊重员工权益,助力企业数字化发展。
35 4
|
4月前
|
存储 算法 C++
C++提高篇:泛型编程和STL技术详解,探讨C++更深层的使用
文章详细探讨了C++中的泛型编程与STL技术,重点讲解了如何使用模板来创建通用的函数和类,以及模板在提高代码复用性和灵活性方面的作用。
69 2
C++提高篇:泛型编程和STL技术详解,探讨C++更深层的使用
|
3月前
|
存储 编译器 C++
【C++篇】引领C++模板初体验:泛型编程的力量与妙用
【C++篇】引领C++模板初体验:泛型编程的力量与妙用
58 9
|
3月前
|
编译器 C语言 C++
C++入门6——模板(泛型编程、函数模板、类模板)
C++入门6——模板(泛型编程、函数模板、类模板)
76 0
C++入门6——模板(泛型编程、函数模板、类模板)
|
4月前
|
存储 算法 安全
超级好用的C++实用库之sha256算法
超级好用的C++实用库之sha256算法
168 1
|
3月前
|
算法 数据处理 C++
c++ STL划分算法;partition()、partition_copy()、stable_partition()、partition_point()详解
这些算法是C++ STL中处理和组织数据的强大工具,能够高效地实现复杂的数据处理逻辑。理解它们的差异和应用场景,将有助于编写更加高效和清晰的C++代码。
55 0
|
3月前
|
存储 算法 程序员
迪杰斯特拉(Dijkstra)算法(C/C++)
迪杰斯特拉(Dijkstra)算法(C/C++)
下一篇
开通oss服务