这两天忙着在准备3月份打PAT考试,许久没有接触刷题了,各种生疏各种忘记,刷题速度那是一个慢,真是为自己智商着急。今天刷题碰到了一个有意思的编程习惯性错误,好几道题都涉及到自定义排序,需要自己重写<操作符号,我在调试的时候,程序提示assertion error: Invalid operator<,思考了很久也没有发现问题,在网上搜寻了这个问题,大概是理解了,将我对这个问题的理解总结一下。
案例:如自定义结构体Cmp
struct Cmp { int field1,field2; bool operator<(const Cmp& cmp) { if(this->field1 < cmp.field1) return true; else if(this->field2 < cmp.field2) return true; return false; } };
此结构体的大小主要是按照field1进行排列,如果field1排不出来,就按照field2进行排列。这段代码看上去没有什么问题,但是如果使用algorithm stl中的sort函数对含有此类型的数组进行排序,在debug模式下会出现断言失败的提示,即invalid operator<,表示这个自定义的比较函数不严格,可能会有问题。
仔细分析,也很容易看出来这段代码错在什么地方,在第二个if判断中,我们仅仅是判断了第二个field大小,忘记加上对第一个field的限制了,把该if判断改成
else if(this->field2 < cmp.field2 && this->field1 == cmp.field1)
发现警告消失了,为什么会这样呢?标准库中的sort函数为什么要我们这么写?
sort函数要求比较函数是strict weak ordering的,而strict weak ordering必须满足三个条件:
1) Strict: pred (X, X) is always false. X跟X自己比为false
2) Weak: If ! pred (X, Y) && !pred (Y, X), X==Y. 当X<Y和Y<X都不成立时,X等于Y
3)Ordering: If pred (X, Y) && pred (Y, Z), then pred (X, Z). 当X<Y,Y<Z时,X<Z成立,即排序的一个传递性。
用一个实例来说明问题。假设有三个变量,m1(1,2),m2(2,1),m3(1,2)。(括号内的两个数分别表示field1和field2的值。在前面有问题的代码下,我们可以判断:
1)m1<m2,并且m2<m3,根据条件3,m1<m3,但是很明显m1跟m3是一样的,应该是相等的!
2)也可以发现在该规则下,m2<m1也成立!根据条件1,推导出m1竟然与m2相等,m1==m2!
这两个情况明显不符合常理,如果允许程序这样运行,最后的结果谁也想不到会排成什么样子。
综上所述,在写比较函数的时候一定要小心,不要粗心大意漏掉了条件。这也告诫了我们编译器给出的各种提示警告,还是老老实实地去遵守比较好,不然到后来吃了哑巴亏也不知道是什么地方出了问题。
相关推荐
invalid type assertion error(解决方案).md
【Navicate Premium 12 64bit】是一款专为Windows操作系统设计的高效能数据库管理工具,尤其适用于64位环境。Navicat是著名的数据库管理和开发软件,它集成了多种数据库系统的连接功能,包括MySQL、Oracle、SQL ...
invalid type assertion(解决方案).md
invalid type assertion syntax(解决方案).md
Generic syntax highlighter syntaxerror: invalid syntax syntaxerror: invalid syntax syntaxerror: invalid syntax syntaxerror: invalid syntax syntaxerror: invalid syntax
syntaxerror: invalid syntax syntaxerror: invalid syntax syntaxerror: invalid syntax syntaxerror: invalid syntax syntaxerror: invalid syntax
std::cout << "Invalid operator!" << std::endl; } ``` 3. 输出结果:计算完成后,使用`std::cout`将结果输出到屏幕上。 ```cpp std::cout << "Result: " << result << std::endl; ``` 4. 错误处理:对于可能...
在Oracle数据库系统中,"invalid rowid"错误通常表示尝试访问的数据行的引用已经失效或者不存在。RowID是Oracle数据库中用于唯一标识表中每一行的一个物理地址,它由数据库内部生成并存储在数据块中。当遇到"invalid...
在Python开发过程中,遇到"error: invalid command ‘egg-info‘"这样的报错通常是由于安装或配置Setuptools库时出现了问题。Setuptools是Python项目构建、安装和管理的工具,而`egg-info`命令用于生成项目的元数据...
### 错误信息解析 在ROS(Robot Operating System)开发过程中,遇到`rlexception: invalid roslaunch xml syntax: no element found: line 1, column 0`这样的错误信息时,表明当前的`.launch`文件存在XML语法错误...
新版启动calico所需镜像quay.io/tigera/operator:v1.20.3
### 解决pip install xxx报错SyntaxError: invalid syntax的问题 #### 背景与问题描述 在使用Python进行开发的过程中,经常会遇到需要安装各种第三方库的情况。这些库可以通过`pip`这一工具轻松安装。然而,在实际...
cout << "Error: Invalid operator!" << endl; break; } return 0; } ``` 五、心得体会和参考资料 通过本课程设计,我们学习了 C++ 语言的基本语法和编程技术,同时也了解了计算器的设计和实现过程。在今后的...
@ERROR: invalid uid nobody rsync error: error starting client-server protocol (code 5) at main.c(1506) [Receiver=3.0.7] 解决办法:在rsyncd.conf文件中添加下面两行即可解决问题 uid =
if (sock == INVALID_SOCKET) { cout << "Error at socket(): " << WSAGetLastError() << endl; WSACleanup(); return 1; } // 设置为混杂模式 DWORD dwBufferInLen = 1; if (WSAIoctl(sock, IO_RCVALL, &...
Delphi 10.3 ,程序运行到DataModule2.ClientDataSet1.ApplyUpdates(0);语句时,提示“Invalid parameter”错误
Invalid Multibyte Character Sequence 警告解析 在编程中,特别是在嵌入式系统开发中,我们经常会遇到Invalid Multibyte Character Sequence 警告。这个警告通常来自于编译器,告知我们存在非法的多字节字符序列。...
qDebug() << "Invalid JSON data"; } else { QJsonObject jsonObject = doc.object(); // 访问JSON对象的键值 QString value1 = jsonObject.value("key1").toString(); QString value2 = jsonObject.value(...
常见的信道估计算法包括最小均方误差(Minimum Mean Square Error, MMSE)和基于训练序列的方法,如最小二乘递归(Recursive Least Squares, RLS)和快速傅里叶变换(Fast Fourier Transform, FFT)为基础的估计算法...