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++


相关文章
|
19天前
|
编译器 C++
C++ 类构造函数初始化列表
构造函数初始化列表以一个冒号开始,接着是以逗号分隔的数据成员列表,每个数据成员后面跟一个放在括号中的初始化式。
60 30
|
8天前
|
并行计算 Unix Linux
超级好用的C++实用库之线程基类
超级好用的C++实用库之线程基类
14 4
|
8天前
|
C++ Windows
HTML+JavaScript构建C++类代码一键转换MASM32代码平台
HTML+JavaScript构建C++类代码一键转换MASM32代码平台
|
8天前
|
C++
2合1,整合C++类(Class)代码转换为MASM32代码的平台
2合1,整合C++类(Class)代码转换为MASM32代码的平台
|
8天前
|
存储 运维 监控
超级好用的C++实用库之日志类
超级好用的C++实用库之日志类
14 0
|
1月前
|
存储 编译器 C++
C ++初阶:类和对象(中)
C ++初阶:类和对象(中)
|
2月前
|
存储 安全 编译器
【C++】类和对象(下)
【C++】类和对象(下)
【C++】类和对象(下)
|
1月前
|
C++
C++(十六)类之间转化
在C++中,类之间的转换可以通过转换构造函数和操作符函数实现。转换构造函数是一种单参数构造函数,用于将其他类型转换为本类类型。为了防止不必要的隐式转换,可以使用`explicit`关键字来禁止这种自动转换。此外,还可以通过定义`operator`函数来进行类型转换,该函数无参数且无返回值。下面展示了如何使用这两种方式实现自定义类型的相互转换,并通过示例代码说明了`explicit`关键字的作用。
|
1月前
|
存储 设计模式 编译器
C++(十三) 类的扩展
本文详细介绍了C++中类的各种扩展特性,包括类成员存储、`sizeof`操作符的应用、类成员函数的存储方式及其背后的`this`指针机制。此外,还探讨了`const`修饰符在成员变量和函数中的作用,以及如何通过`static`关键字实现类中的资源共享。文章还介绍了单例模式的设计思路,并讨论了指向类成员(数据成员和函数成员)的指针的使用方法。最后,还讲解了指向静态成员的指针的相关概念和应用示例。通过这些内容,帮助读者更好地理解和掌握C++面向对象编程的核心概念和技术细节。
|
2月前
|
存储 算法 编译器
c++--类(上)
c++--类(上)
下一篇
无影云桌面