记录线上APP一个排序比较引发的崩溃 Comparison method violates its general contract!
最近在做产品需求的时候上线了一个新的产品需求,给用户多了一种新的排序排序规则,更加方便用户找到自己想要的东西。新版本发布后,QA 给我发了一个 线上崩溃 bug 链接,具体内容如下:

看到上面的链接,我有点懵逼了,就这排序还能给我搞出 bug 来?看到抛出的异常信息,也没有见过,于是直接百度搜索了。
一百度,发现很多人遇到这个问题,下面简单说下出现这个问题的原因:
在 JDK7 版本以上,Comparator 要满足自反性,传递性,对称性,不然 Arrays.sort,Collections.sort
会报 IllegalArgumentException 异常。
自反性:当 两个相同的元素相比时,compare必须返回0,也就是compare(o1, o1) = 0;
反对称性:如果compare(o1,o2) = 1,则compare(o2, o1)必须返回符号相反的值也就是 -1;
传递性:如果 a>b, b>c, 则 a必然大于c。也就是compare(a,b)>0, compare(b,c)>0, 则compare(a,c)>0
相信很多人看到这里还是会很懵逼的,感觉自己写的代码是不会出现这个问题的,这里理解的主要难点是怎么复现这个崩溃。
任何问题在我们一开始看到的时候,都会觉得很奇怪,觉得自己写的代码是不会出现这种问题的,可是一旦复现后,就会突然顿悟了,还是有自己遗漏没有想到的 case 。
例子
demo1
其实违反上述规则最简单的例子就是如下:
new Comparator<Student>() {
@Override
public int compare(Student o1, Student o2) {
return o1.getId() > o2.getId() ? 1 : -1;
}
}
出现原因:没有考虑相等的情形,所以会抛出异常。
不过对于有基础的程序猿,一般都会考虑到等号的情形,所以上述代码还是很少会出现的。
new Comparator<Student>() {
@Override
public int compare(Student o1, Student o2) {
if (o1.getId() == o2.getId())
return 0;
return o1.getId() > o2.getId() ? 1 : -1;
}
}
如果按照上面的来,基本就不会有问题了。当然有个点需要注意的是需要判空。
不过我的崩溃和上面的例子还是很不一样的,下面举一个特殊的例子。
demo 2 (线上崩溃例子)
相信大家都用过手机的通讯录,我手机通讯录的排序方式是 # AB...YZ 这种形式的。也就是按照用户名来进行排序的,非字母类型的需要排在前面。
我出代码的问题其实就出现在对于 # 这一类名字的处理。下面看错误代码:
private static Comparator<CompareObject> mComparatorByPlayingAndLetter = new Comparator<CompareObject>() {
@Override
public int compare(CompareObject o1, CompareObject o2) {
char firstChar = o1.name.charAt(0);
char secondChar = o2.name.charAt(0);
if (!isUpperLetters(firstChar) && (isUpperLetters(secondChar))) {
return 1;
}
if (isUpperLetters(firstChar) && !isUpperLetters(secondChar)) {
return -1;
}
if (!isUpperLetters(firstChar) && (!isUpperLetters(secondChar))) {
return 1;
}
return o1.name.compareToIgnoreCase(o2.name);
}
};
这里我先说下自己排序的算法思想:
如果一个是大写字母,一个是非大写字母,那么很好排序;
如果两个都是非大写字母,我返回1或者-1都可以,这里我直接给了1,对于非大写字母后面和大写字母的比较,前面的逻辑会进行处理,剩下的就是大写字母之间的比较了。
本地测试,没问题的。QA 测试也是没问题。然后这段代码上线了。
结果昨天刚发布正式版,今天就收到 QA 抛过来的线上崩溃,不过还好只是一个崩溃量。但是为啥会崩溃,我还是没法理解,我本地测试了很多遍,也还是无法复现。也百度看了很多文章,虽然知道崩溃的理论原因,但是如果无法复现,我就还是不能理解。
并且虽然我有崩溃用户的 cuid,但是崩溃的用户的数据排序我是没法拿到的,也就是还是无法复现。后来自己在已有的数据中,加了一些特殊的字符后,终于复现了。
下面来看一下 ASCII 字符表:

可以看到的是 AB...YZ 是处于后半部分的,数字和大部分特殊符号都是在大写字母前面,然后有部分标点符号是在大写字母后面的。
于是,我利用原有的数据,然后再在其中加入大写字母前后的特殊字符。对于这些数据,除了我这次新增的排序,还有其他排序,比如字母排序,创建时间排序等,不断对这些数据采用其他排序进行展示,然后再切到出问题的排序,多次来回切换排序算法,最终复现了该问题。

但是具体是哪些数据排序后引起的不满足规则,由于数据量比较大,我无法确定出来。但是可以知道的是,最后引起崩溃的两个名字只是雪花,真正有问题的地方在出现问题前就已经埋下了。
那对于上面的问题,如何解决呢?
private static Comparator<CompareObject> mComparatorByPlayingAndLetter = new Comparator<CompareObject>() {
@Override
public int compare(CompareObject o1, CompareObject o2) {
char firstChar = o1.name.charAt(0);
char secondChar = o2.name.charAt(0);
if (!isUpperLetters(firstChar) && (isUpperLetters(secondChar))) {
return 1;
}
if (isUpperLetters(firstChar) && !isUpperLetters(secondChar)) {
return -1;
}
if (!isUpperLetters(firstChar) && (!isUpperLetters(secondChar))) {
return 1;
} // 删除红色代码即可
return o1.name.compareToIgnoreCase(o2.name);
}
};
总之,以后再写排序比较的时候,对于无法确定大小的情况,交给系统的排序,不要自己去随意改变比较值,这样就不会出现这种 case 了。
记录线上APP一个排序比较引发的崩溃 Comparison method violates its general contract!的更多相关文章
- 排序遇到问题 JDK7的Comparison method violates its general contract
图解JDK7的Comparison method violates its general contract异常 楼主分析的很详细,能力有限,我看得迷迷糊糊的,不过大致知道这个错误的起因了.学习了,谢 ...
- 关于jdk7中 使用Collections的排序方法时报Comparison method violates its general contract!异常
参考: Comparison method violates its general contract Comparison method violates its general contract! ...
- 记录线上与本地docker镜像一致,但Dockerfile却构建失败的问题
背景 公司新开了某个项目,我在新的服务器部署了docker环境,本着ctrl+c 和ctrl+v的惯例,直接把以前的php环境的Dockerfile文件直接复制到新项目服务器那里,结果构建失败,失败的 ...
- Jedis线上的一个小坑:Redis有并发访问的数据错乱的问题
问题现象: 业务数据有错乱,A的一些数据有好几个都是B的数据 这些业务数据在保存在Redis缓存中,怀疑是并发情况下Jedis错乱的问题 原因分析: JedisUtil里面在使用完Jedis 后释放资 ...
- 记录线上一次线程hang住问题
线上发现执行某特定任务在某个特定时间点后不再work.该任务由线程池中线程执行定时周期性调度,根据日志查看无任何异常.从代码研判应该无关定时任务框架,因为对提交的定时任务做了wrap,会将异常都cat ...
- 火热的线上APP的源码分享,开箱即用
这篇文章是写给iOS的程序员或产品经理的,同样,对于入门学习iOS开发的人,也是一个很好的实战演练,因为这里分享的是一个已经上架的.拿了源码就能正常运行起来的项目. 在介绍这个项目的源码分享之前,小编 ...
- 【Redis连接超时】记录线上RedisConnectionFailureException异常排查过程
项目架构: 部分组件如下: SpringCloudAlibaba(Nacos+Gateway+OpenFeign)+SpringBoot2.x+Redis 问题背景: 最近由于用户量增大,在高峰时期, ...
- 如何有效的跟踪线上 MySQL 实例表和权限的变更
介绍 从系统管理员或 DBA 的角度来讲, 总期望将线上的各种变更限制在一个可控的范围内, 减少一些不确定的因素. 这样做有几点好处: . 记录线上的库表变更; . 对线上的库表变更有全局的了解; . ...
- TFS线上生成环境发布历程
继前文 TFS在项目中Devops落地进程(上) TFS在项目中DevOps落地进程(下) 自从之前将开发环境使用TFS进行了自动化之后,就享受在此成果中,其他后续进度就停顿了好一段时间. 毕竟在我们 ...
随机推荐
- CF1511G-Chips on a Board【倍增】
正题 题目链接:https://www.luogu.com.cn/problem/CF1511G 题目大意 给出\(n*m\)的棋盘上每一行有一个棋子,双方轮流操作可以把一个棋子向左移动若干步(不能不 ...
- AT4519-[AGC032D]Rotation Sort【dp】
正题 题目链接:https://www.luogu.com.cn/problem/AT4519 题目大意 给出一个长度为\(n\)的排列,每次可以选择一个区间,然后花费\(A\)的代价向左旋转(最左边 ...
- 构建idea父工程
构建idea父工程 首先通过idea新建一个Maven项目: 选择本地Maven版本: 工程名称: 选择字符编码:utf-8 file -> Settings -> Editor -> ...
- pure-ftpd管理FTP服务器,创建文件夹可以,但上传下载文件不行
两种原因 1.因为pure-ftpd的防火墙端口问题 # Port range for passive connections replies. - for firewalling. PassiveP ...
- ubuntu修改软件源的方法
最快方法--替换法 刚安装好的ubutun,打开source.list后,用vim替换的方法将所有的us提付出替换为 cn,然后保存退出,更新即可. # vim /etc/apt/source.lis ...
- 乘风破浪,遇见最美Windows 11之新微软商店(Microsoft Store)生态 - 安卓(Android™)开发体验指南
什么是Windows 11的安卓(Android)应用 2021年6月25日,微软召开线上发布会,对外宣告下一代Windows操作系统Windows 11,Windows 11为用户重新打造的Micr ...
- javascriptRemke之类的继承
前言:es6之前在js中要实现继承,就必须要我们程序员在原型链上手动继承多对象的操作,但是结果往往存在漏洞,为解决这些问题,社区中出现了盗用构造函数.组合继承.原型式继承.寄生式继承等一系列继承方式, ...
- ZK(ZooKeeper)分布式锁实现
点赞再看,养成习惯,微信搜索[牧小农]关注我获取更多资讯,风里雨里,小农等你. 本文中案例都会在上传到git上,请放心浏览 git地址:https://github.com/muxiaonong/Zo ...
- CentOS 文件管理
目录 目录管理 目录结构 切换目录 查看目录 创建目录 复制目录 剪切目录 删除目录 文件管理 查看文件 创建文件 复制文件 剪切文件 删除文件 创建链接 目录管理 目录也是一种文件. 蓝色目录,绿色 ...
- Kruskal重构树-进阶
例题一:区间最小生成树(NKOJ P8439) 简要题意: 一个n个点m条边的无向图,点编号1到n,边编号1到m.边有边权. 有q次操作,操作分两种: 1.k x y z:修改第k条边,使其连接的两点 ...