两个List比较内容是否一样
来源:互联网 发布:苹果日历群发软件 编辑:程序博客网 时间:2024/06/05 14:13
刚刚看到CSDN上一句标语:编程之久除了算法和数据结构,什么也不属于我们。
我觉得这句话有一定的道理。语言在变,技术在变,各种技术标准也终究会变,我们唯一能跟得上变化的是编程思想,而编程思想中很重要的两个方面就是算法和数据结构。除此之外我觉得设计思想比如设计模式也很重要。先记下,以后慢慢再领悟。
不知道代码块上传之后自己怎么查看,所以先记在博客里。
实现
一段进行对两个list进行比较的代码。
/** * 首先进行入参检查防止出现空指针异常 * 如果两个参数都为空,则返回true * 如果有一项为空,则返回false * 接着对第一个list进行遍历,如果某一项第二个list里面没有,则返回false * 还要再将两个list反过来比较,因为可能一个list是两一个list的子集 * 如果成功遍历结束,返回true * @param l0 * @param l1 * @return */public static boolean isListEqual(List l0, List l1){ if (l0 == l1) return true; if (l0 == null && l1 == null) return true; if (l0 == null || l1 == null) return false; if (l0.size() != l1.size()) return false; for (Object o : l0) { if (!l1.contains(o)) return false; } for (Object o : l1) { if (!l0.contains(o)) return false; } return true; }
原理
这种比较可以成功比较两个内容相同但是元素顺序不同的list。
比较关键的是使用了List.contains()方法,这个方法是个接口。以ArrayList为例,看一下ArrayList里的实现:
/** * Returns <tt>true</tt> if this list contains the specified element. * More formally, returns <tt>true</tt> if and only if this list contains * at least one element <tt>e</tt> such that * <tt>(o==null ? e==null : o.equals(e))</tt>. * * @param o element whose presence in this list is to be tested * @return <tt>true</tt> if this list contains the specified element */ public boolean contains(Object o) { return indexOf(o) >= 0; } /** * Returns the index of the first occurrence of the specified element * in this list, or -1 if this list does not contain the element. * More formally, returns the lowest index <tt>i</tt> such that * <tt>(o==null ? get(i)==null : o.equals(get(i)))</tt>, * or -1 if there is no such index. */ public int indexOf(Object o) { if (o == null) { for (int i = 0; i < size; i++) if (elementData[i]==null) return i; } else { for (int i = 0; i < size; i++) if (o.equals(elementData[i])) return i; } return -1; }
contains()返回的是indexOf(Object o)是否大于0的boolean值。indexOf(Object o)返回的结果是目标对象的下标值。
elementData是ArrayList里的Object的数组,elementData[]的声明(为什么将类型定为transient 不太明白):
// non-private to simplify nested class accesstransient Object[] elementData;
首先看indexOf()方法,当目标Object为空时,返回elementData中第一个为空元素的下标。否则如果目标Object不为空,调用Object的equals()方法,返回第一个相等元素的下标。
当上述两种情况都能找到相应元素时,下标肯定都是大于等于0的,如果没找到的话返回-1。
因此当contains()只需判断indexOf()的返回值是否大于等于0就可以判断该list是否包含目标元素。
问题
但是这种简单的判断两个list内容是否一样的代码有一个问题,因为最终的元素的比较依赖于equals()方法。当List里的元素没有重写equals()方法时,这个判断的结果是不准确的。
class AA{ public String s; public AA(String s){ this.s = s; }}ArrayList l0 = new ArrayList();l0.add(new AA("aaa"));l0.add(new AA("bbb"));ArrayList l1 = new ArrayList();l1.add(new AA("aaa"));l1.add(new AA("bbb"));System.out.println(isListEqual(l0, l1));
上述代码应该返回的结果为true,但实际返回的结果为false。
class AA{ public String s; public AA(String s){ this.s = s; } @Override public boolean equals(Object obj) { if (this == null && obj == null) return true; if (this == null || obj == null) return false; if (obj.getClass() != this.getClass()) return false; AA a = (AA) obj; if (this.s.equals(a.s)) return true; return false; }}
为AA类重写equals()后再次运行返回结果正常。
结论
1.凡涉及到对象的比较是否相同时,确保实现了equals()方法,以免发生意外。
2.该判断方法的代码只针对包含了实现了equals的对象的list。
未解决的问题
ArrayList里的elementData字段为什么要加transient 关键字
- 两个list比较内容是否一样:disjoint
- 两个List比较内容是否一样
- java 比较两个无序list中的值是否一样
- iOS中比较两个数组是否一样
- mongo 比较两个字段值是否一样
- java判断两个文件的内容是否完全一样
- Oracle PL/SQL比较两个字段是否一样
- 比较组成两个字符串的字符是否一样
- linux比较两个文件是否一样(linux命令md5sum使用方法)
- linux比较两个文件是否一样(linux命令md5sum使用方法)
- linux之使用md5sum命令比较两个文件是否一样
- 比较两个 List 的值是否相等
- 比较两个list集合是否相等
- 判断两个对象是否一样
- C#如何实现比较两个文件的内容是否完全相同?
- Java课程设计 比较两个文件内容是否相同
- 比较两个一维数组的内容是否相等
- VC中比较两个文件是否内容相等
- 数据库事务
- 利用pstack 和 strace分析程序在哪里耗时?
- Kubernetes Https证书转换方法
- The study of generator in Python(20170912)
- 梯度下降法的matlab实现
- 两个List比较内容是否一样
- JavaScript跨域与解决方案详解
- 联想ThinkPad系列笔记本进bios设置u盘启动教程
- 业余草推荐一款局域网(内网)穿透工具lanproxy
- Altium Designer 导入Arduino UNO PCB
- JSONP 在前端的发送和后台node.js的处理
- HDU 1160 FatMouse's Speed
- 用量子计算辅助深度学习:研究者提出量子辅助Helmholtz机
- Django之URL的命名空间和命名模式