C++中自定义结构体或类作为关联容器的键

简介: C++中自定义结构体或类作为关联容器的键

C++中自定义结构体或类作为关联容器的键

目录

1. 概述

STL中像set和map这样的容器是通过红黑树来实现的,插入到容器中的对象是顺序存放的,采用这样的方式是非常便于查找的,查找效率能够达到O(log n)。所以如果有查找数据的需求,可以采用set或者map。

但是我们自定义的结构体或者类,无法对其比较大小,在放入到容器中的时候,就无法正常编译通过,这是set/map容器的规范决定的。要将自定义的结构体或者类存入到set/map容器,就需要定义一个排序的规则,使其可以比较大小。最简单的办法就是在结构体或者类中加入一个重载小于号的成员函数,这样在存数据进入set/map中时,就可以根据其规则排序。

2. 实例

在这里就写了一个简单的例子,将自定义的一个二维点存入set/map,并查找其中存入的数据:

#include <iostream>
#include <map>
#include <set>
#include <string>
using namespace std;
const double EPSILON = 0.000001;
// 2D Point
struct Vector2d
{
public:
  Vector2d()
  {
  }
  Vector2d(double dx, double dy)
  {
    x = dx;
    y = dy;
  }
  // 矢量赋值
  void set(double dx, double dy)
  {
    x = dx;
    y = dy;
  }
  // 矢量相加
  Vector2d operator + (const Vector2d& v) const
  {
    return Vector2d(x + v.x, y + v.y);
  }
  // 矢量相减
  Vector2d operator - (const Vector2d& v) const
  {
    return Vector2d(x - v.x, y - v.y);
  }
  //矢量数乘
  Vector2d Scalar(double c) const
  {
    return Vector2d(c*x, c*y);
  }
  // 矢量点积
  double Dot(const Vector2d& v) const
  {
    return x * v.x + y * v.y;
  }
  //向量的模
  double Mod() const
  {
    return sqrt(x * x + y * y);
  }
  bool Equel(const Vector2d& v) const
  {
    if (abs(x - v.x) < EPSILON && abs(y - v.y) < EPSILON)
    {
      return true;
    }
    return false;
  }
  bool operator == (const Vector2d& v) const
  {
    if (abs(x - v.x) < EPSILON && abs(y - v.y) < EPSILON)
    {
      return true;
    }
    return false;
  }
  bool operator < (const Vector2d& v) const
  {   
    if (abs(x - v.x) < EPSILON)
    {
      return y < v.y ? true : false;
    }
    return x<v.x ? true : false;
  }
  double x, y;
};
int main()
{ 
  {
    set<Vector2d> pointSet; 
    pointSet.insert(Vector2d(0, 11));
    pointSet.insert(Vector2d(27, 63));
    pointSet.insert(Vector2d(27, 15));
    pointSet.insert(Vector2d(0, 0));
    pointSet.insert(Vector2d(67, 84));
    pointSet.insert(Vector2d(52, 63));
    for (const auto &it : pointSet)
    {
      cout << it.x << '\t' << it.y << endl;
    }
    auto iter = pointSet.find(Vector2d(27, 63));
    if (iter == pointSet.end())
    {
      cout << "未找到点" << endl;
    }
    else
    {
      cout << "可以找到点" << endl;
    }
  } 
  {
    map<Vector2d, string> pointSet;
    pointSet.insert(make_pair(Vector2d(52, 63), "插入时的第1个点"));
    pointSet.insert(make_pair(Vector2d(27, 63), "插入时的第2个点"));
    pointSet.insert(make_pair(Vector2d(0, 11), "插入时的第3个点"));    
    pointSet.insert(make_pair(Vector2d(67, 84), "插入时的第4个点"));
    pointSet.insert(make_pair(Vector2d(27, 15), "插入时的第5个点"));
    pointSet.insert(make_pair(Vector2d(0, 0), "插入时的第6个点"));
  
    for (const auto &it : pointSet)
    {
      cout << it.first.x << ',' << it.first.y << '\t' << it.second << endl;
    }
    auto iter = pointSet.find(Vector2d(27, 63));
    if (iter == pointSet.end())
    {
      cout << "未找到点" << endl;
    }
    else
    {
      cout << "可以找到点" << endl;
    }
  }
}

其中的关键就是在点的结构体中重载了<符号的比较函数,规定首先比较y的大小,其次在比较x的大小:

bool operator < (const Vector2d& v) const
{   
    if (abs(x - v.x) < EPSILON)
    {
        return y < v.y ? true : false;
    }
    return x<v.x ? true : false;
}

最终的运行结果如下:

分类: C++

标签: STL , C++


相关文章
|
20天前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
30 2
|
26天前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
67 5
|
1月前
|
存储 编译器 C++
【c++】类和对象(中)(构造函数、析构函数、拷贝构造、赋值重载)
本文深入探讨了C++类的默认成员函数,包括构造函数、析构函数、拷贝构造函数和赋值重载。构造函数用于对象的初始化,析构函数用于对象销毁时的资源清理,拷贝构造函数用于对象的拷贝,赋值重载用于已存在对象的赋值。文章详细介绍了每个函数的特点、使用方法及注意事项,并提供了代码示例。这些默认成员函数确保了资源的正确管理和对象状态的维护。
70 4
|
1月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
80 4
|
22天前
|
存储 设计模式 C++
【C++】优先级队列(容器适配器)
本文介绍了C++ STL中的线性容器及其适配器,包括栈、队列和优先队列的设计与实现。详细解析了`deque`的特点和存储结构,以及如何利用`deque`实现栈、队列和优先队列。通过自定义命名空间和类模板,展示了如何模拟实现这些容器适配器,重点讲解了优先队列的内部机制,如堆的构建与维护方法。
31 0
|
2月前
|
存储 编译器 对象存储
【C++打怪之路Lv5】-- 类和对象(下)
【C++打怪之路Lv5】-- 类和对象(下)
30 4
|
2月前
|
编译器 C语言 C++
【C++打怪之路Lv4】-- 类和对象(中)
【C++打怪之路Lv4】-- 类和对象(中)
26 4
|
2月前
|
存储 安全 C++
【C++打怪之路Lv8】-- string类
【C++打怪之路Lv8】-- string类
25 1
|
2月前
|
存储 编译器 C++
【C++类和对象(下)】——我与C++的不解之缘(五)
【C++类和对象(下)】——我与C++的不解之缘(五)
|
2月前
|
编译器 C++
【C++类和对象(中)】—— 我与C++的不解之缘(四)
【C++类和对象(中)】—— 我与C++的不解之缘(四)