C++容器map及unordered_map底层实现和性能上有所不同

std::map 和 std::unordered_map都是 C++ 标准库中用于实现关联容器的类模板,但它们在底层实现和性能上有所不同。

1. std::map:

  • 使用红黑树实现,保持元素的有序性。因此,所有操作的时间复杂度都是 O(log n),其中 n 是元素个数。

  • 元素按键排序存储,键是唯一的,每个键最多只能出现一次。

  • 适用于需要有序存储键值对的情况,例如需要按照键的顺序进行遍历或查找的场景。

当需要按照键的顺序进行存储和访问时,可以使用 std::map。例如,存储学生的分数,并按照学生姓名的字典顺序进行排序:

cpp 复制代码
#include <iostream>
#include <map>
#include <string>

int main() {
    std::map<std::string, int> studentScores;

    // 添加学生分数
    studentScores["Alice"] = 95;
    studentScores["Bob"] = 87;
    studentScores["Charlie"] = 92;

    // 遍历输出学生分数(按姓名字典顺序)
    for (const auto& pair : studentScores) {
        std::cout << pair.first << ": " << pair.second << std::endl;
    }

    return 0;
}

2. std::unordered_map:

  • 使用哈希表实现,插入、查找、删除操作的平均时间复杂度为 O(1),但最坏情况下可能会达到 O(n)。

  • 元素存储时没有顺序限制,但键必须是唯一的。

  • 适用于对插入和查找性能有较高要求,并且不要求元素有序存储的情况。

因此,选择 std::map 还是 std::unordered_map 取决于具体的需求。如果需要有序存储或者操作少量数据,可以选择 std::map;如果对性能要求较高,并且不需要有序存储,可以选择 std::unordered_map。

cpp 复制代码
#include <iostream>
#include <unordered_map>
#include <string>

int main() {
    std::unordered_map<std::string, int> studentScores;

    // 添加学生分数
    studentScores["S001"] = 95;
    studentScores["S002"] = 87;
    studentScores["S003"] = 92;

    // 查找学生分数
    std::string studentID = "S002";
    if (studentScores.find(studentID) != studentScores.end()) {
        std::cout << "Score of student " << studentID << ": " << studentScores[studentID] << std::endl;
    } else {
        std::cout << "Student " << studentID << " not found." << std::endl;
    }

    return 0;
}

**`std::unordered_map` 提供了一系列方法来进行元素的插入、查找、删除等操作。**以下是一些常用的方法:

  1. **插入元素**:
  • `insert`: 将键值对插入到 `std::unordered_map` 中。
  1. **访问元素**:
  • `operator[]`: 通过键来访问元素的值。

  • `at`: 通过键来访问元素的值,类似于 `operator[]`,但提供了边界检查。

  1. **查找元素**:
  • `find`: 查找给定键的元素,返回指向该元素的迭代器,如果未找到,则返回 `std::unordered_map::end()`。

  • `count`: 返回指定键的元素个数,由于每个键在 `std::unordered_map` 中最多出现一次,因此结果为 0 或 1。

  1. **删除元素**:
  • `erase`: 删除指定键的元素或一定范围内的元素。

  • `clear`: 清空 `std::unordered_map` 中的所有元素。

  1. **容量相关**:
  • `empty`: 检查 `std::unordered_map` 是否为空。

  • `size`: 返回 `std::unordered_map` 中元素的数量。

  1. **哈希策略**:
  • `hash_function`: 返回哈希函数对象,用于计算键的哈希值。

  • `key_eq`: 返回用于比较键是否相等的函数对象。

这些方法使得在 `std::unordered_map` 中进行元素的操作变得非常方便。


**`std::map` 提供了一系列方法来进行元素的插入、查找、删除等操作。**以下是一些常用的方法:

  1. **插入元素**:
  • `insert`: 将键值对插入到 `std::map` 中。
  1. **访问元素**:
  • `operator[]`: 通过键来访问元素的值。

  • `at`: 通过键来访问元素的值,类似于 `operator[]`,但提供了边界检查。

  1. **查找元素**:
  • `find`: 查找给定键的元素,返回指向该元素的迭代器,如果未找到,则返回 `std::map::end()`。

  • `count`: 返回指定键的元素个数,由于每个键在 `std::map` 中最多出现一次,因此结果为 0 或 1。

  1. **删除元素**:
  • `erase`: 删除指定键的元素或一定范围内的元素。

  • `clear`: 清空 `std::map` 中的所有元素。

  1. **容量相关**:
  • `empty`: 检查 `std::map` 是否为空。

  • `size`: 返回 `std::map` 中元素的数量。

  1. **迭代器**:
  • `begin`: 返回指向第一个元素的迭代器。
  • `end`: 返回指向最后一个元素之后位置的迭代器。
  1. **比较操作**:
  • `key_comp`: 返回用于比较键的函数对象。

  • `value_comp`: 返回用于比较值的函数对象。

这些方法使得在 `std::map` 中进行元素的操作变得非常方便。

相关推荐
小坏坏的大世界1 小时前
C++ STL常用容器总结(vector, deque, list, map, set)
c++·算法
wjs20242 小时前
状态模式(State Pattern)
开发语言
我命由我123452 小时前
Kotlin 数据容器 - List(List 概述、创建 List、List 核心特性、List 元素访问、List 遍历)
java·开发语言·jvm·windows·java-ee·kotlin·list
liulilittle2 小时前
C++ TAP(基于任务的异步编程模式)
服务器·开发语言·网络·c++·分布式·任务·tap
励志要当大牛的小白菜4 小时前
ART配对软件使用
开发语言·c++·qt·算法
PAK向日葵5 小时前
【算法导论】如何攻克一道Hard难度的LeetCode题?以「寻找两个正序数组的中位数」为例
c++·算法·面试
爱装代码的小瓶子6 小时前
数据结构之队列(C语言)
c语言·开发语言·数据结构
Maybe_ch7 小时前
.NET-键控服务依赖注入
开发语言·c#·.net
超浪的晨8 小时前
Java UDP 通信详解:从基础到实战,彻底掌握无连接网络编程
java·开发语言·后端·学习·个人开发
终焉暴龙王8 小时前
CTFHub web进阶 php Bypass disable_function通关攻略
开发语言·安全·web安全·php