如何使用set包含类对象的问题

b_horse 2007-02-28 03:47:04
class定义如下:
class A
{
double _a;
string _b;
public:
A(double a, string b) : _a(a), _b(b){}
bool operator<(const A& b) const { return _a < b._a; }
bool operator==(const A& b) const { return _a == b._a; }

friend ostream& operator<<(ostream& os, const A& a)
{
os << a._a << " and " << a._b << endl;
return os;
}
};
定义operator的目的是,想用set容器,以_a值排序,以_b值保持set中元素的唯一性。但好像结果出乎意料:

程序如下:
A a1(1, "hehe");
A a2(2.01, "haha");
A a3(2.01, "hehe");
set<A> a_set;
a_set.insert(a1);
a_set.insert(a3);
a_set.insert(a2);
copy(a_set.begin(), a_set.end(), ostream_iterator<A>(cout, ""));
if (a1 == a3)
{
cout << "a1 == a3" << endl;
}else
{
cout << "a1 != a3" << endl;
}

输出:
1 and hehe
2.01 and hehe
a1 == a3

如果把class中的operator定义改一下:
bool operator<(const A& c) const { return _b < c._b; }
bool operator==(const A& c) const { return _b == c._b; }
则输出为:
2.01 and haha
1 and hehe
a1 == a3

好像,c++要求operator中比较的量相同时才保证标准里定义的behavior的一致性。
不知道有谁知道上面的代码问题在哪里!?
先谢谢了
...全文
375 9 打赏 收藏 转发到动态 举报
写回复
用AI写文章
9 条回复
切换为时间正序
请发表友善的回复…
发表回复
sinall 2007-02-28
  • 打赏
  • 举报
回复
以_a的值排序,以_b的值保证set中值的唯一性
<=>
以_b的值保证set中值的唯一性
<=>
bool operator<(const A& c) const { return _b < c._b; }

不知道楼主何来“_a的值排序”的问题?
b_horse 2007-02-28
  • 打赏
  • 举报
回复
class ACmp
{
public:
bool operator()(const A& left, const A& right) const
{
return left._b!=right._b && left._a < right._a;
}
};
是正解!
jixingzhong 2007-02-28
  • 打赏
  • 举报
回复
set 无法插入重复元素 ~
jixingzhong 2007-02-28
  • 打赏
  • 举报
回复
使用 multiset ......
sinall 2007-02-28
  • 打赏
  • 举报
回复
定义operator的目的是,想用set容器,以_a值排序,以_b值保持set中元素的唯一性。但好像结果出乎意料:
——————————————————————————————————————————
貌似只和operator<相关。另外,不存在所谓排序的问题,只存在保证唯一性的问题。
也就是说,set容器插入元素的时候,根据operator<来判断2个元素是否相同,即:
!(a < b) && !(b < a)

而你的if (a1 == a3)则是调用operator==,二者原理完全不同!
taodm 2007-02-28
  • 打赏
  • 举报
回复
看《Effective STL》item 19
b_horse 2007-02-28
  • 打赏
  • 举报
回复
纠正一下,如果定义是这样的话,(以_a的值排序,以_b的值保证set中值的唯一性)
bool operator<(const A& b) const { return _a < b._a; }
bool operator==(const A& b) const { return _b == b._b ; }

且有
A a1(1, "hehe");
A a2(2.01, "haha");
A a3(2.01, "hehe");
set<A> a_set;
a_set.insert(a1);
a_set.insert(a3);
a_set.insert(a2);

输出为:
1 and hehe
2.01 and hehe
a1 == a3
比较奇怪,因为,a1和a3是相等的,可见==在向set中插值时,并没起作用。

acange 2007-02-28
  • 打赏
  • 举报
回复
你可以修改你的< == operator, 使他比较是也考虑 _b 这个成员.
bool operator<(const A& b) const { return ... }
bool operator==(const A& b) const { return ... }
acange 2007-02-28
  • 打赏
  • 举报
回复
这里问题在于你在class A 中定义:
bool operator<(const A& b) const { return _a < b._a; }
bool operator==(const A& b) const { return _a == b._a; }

你是根据_a (double)的大小来比较A 的对象的大小, 当你定义 a1 a2 a3后,并插入到a_set 中时,你的a2没有被插入, 这是因为a3._a = a2._a. 所以在a_set中只有2 个元素.
A a1(1, "hehe");
A a2(2.01, "haha");
A a3(2.01, "hehe");
set<A> a_set;
a_set.insert(a1);
a_set.insert(a3);
a_set.insert(a2); //a2没有被插入


所以你的输出是这样的.
输出:
1 and hehe
2.01 and hehe
该课程由付强老师主讲,系统的、全面的、具体的讲解了java入门的知识。可以使初级的学员入门。Java入门Java的历史、Java的版本、Java的优势、软件行业前景Java开发环境搭建、编写Java入门练习虚拟机的运行机制、Java的平台无关性、虚拟机垃圾回收机制基础语法关键字、标识符、Java数据类型分类、基础数据类型、取值范围变量、常量、三种注释、生成doc文档、运算符、表达式if语句、switch语句、嵌套判断for语句、while语句、do-while语句、嵌套循环类和对象面向过程和面向对象的区别类的构成、访问修饰符、对象的内存分配this关键字、按值传递与按引用传递、对象的初始化顺序高级类特性类的继承、父子类的初始化顺序、单继承性方法的覆盖、重载、super关键字、多态、instanceof关键字、强制类型转换、static关键字、final关键字数组和枚举一维数组的应用及内存分配多维数组的应用及内存分配数组的复制、数组的按引用传递排序算法:冒泡、直接选择、插入选择、希尔、快速、归并、堆排序Arrays工具类的应用枚举类型的应用常见类的使用Object类的底层及应用、Objects类String类的底层及应用、正则表达式StringBuffer和StringBuilder的底层及应用Math类的应用、包装类的应用日期类的应用:Date、DateFormat、Calendar、LocalDateTime、Instant、LocalDate、MonthDay、ZonedDateTime、YearMonth、新旧日期转换BigInteger和BingDecimal、DecimalFormatSystem类、Scanner类抽象类和接口抽象类的规则及应用接口的规则及应用默认方法、静态方法、函数式接口、Lambda表达式异常异常的定义异常的处理:抓(try-catch-finally)、抛(throws)异常的分类、自定义异常的应用、throw关键字集合框架集合框架结构:接口、实现类Collection接口的方法、Set接口的方法、List接口的方法、Map接口的方法Array、Linked、Hash、Tree底层实现原理泛型的作用、Collections工具类、历史集合类I/O流Path类的原理及应用、Files类的原理及应用文件字节流FileInputStream的原理及应用对象类、缓冲流、数据流的原理及应用字符流的原理及应用多线程进程与线程的概念、查看线程对象Java内存模型线程的创建与启动:扩展Thread类、实现Runnable接口、实现Callable接口、线程池线程状态的转换:新建、就绪、运行、阻塞、死亡线程的调度:sleep、yield、join、interrupt、wait后台线程、定时任务线程的并发与同步、同步锁、同步块、线程安全的类Lock接口、CAS、volatile修饰符内部类成员内部类、本地内部类的应用匿名内部类的原理及引用、Lambda表达式设计模式基础设计模式概念、框架概念7大设计原则:开闭原则、依赖倒置原则、单一职责原则、接口隔离原则、迪米特原则、里氏替换原则、合成复用原则单例模式、工厂模式、模板模式、代理模式、装饰模式、适配器模式、外观模式、策略模式、观察者模式、命令模式、备忘录模式、观察者模式反射反射包Class类的使用反射获取类属性、方法、构造器通过反射创建类对象/通过反射调用方法反射的应用

64,661

社区成员

发帖
与我相关
我的任务
社区描述
C++ 语言相关问题讨论,技术干货分享,前沿动态等
c++ 技术论坛(原bbs)
社区管理员
  • C++ 语言社区
  • encoderlee
  • paschen
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
  1. 请不要发布与C++技术无关的贴子
  2. 请不要发布与技术无关的招聘、广告的帖子
  3. 请尽可能的描述清楚你的问题,如果涉及到代码请尽可能的格式化一下

试试用AI创作助手写篇文章吧