在Java中如何高效的判断数组中是否包含某个元素
如何检查一个数组(无序)是否包含一个特定的值?这是一个在Java中经常用到的并且非常有用的操作。同时,这个问题在Stack Overflow中也是一个非常热门的问题。在投票比较高的几个答案中给出了几种不同的方法,但是他们的时间复杂度也是各不相同的。本文将分析几种常见用法及其时间成本。
检查数组是否包含某个值的方法
使用List
1
2
3
|
public static boolean useList(String[] arr, String targetValue) { return Arrays.asList(arr).contains(targetValue); } |
使用Set
1
2
3
4
|
public static boolean useSet(String[] arr, String targetValue) { Set<String> set = new HashSet<String>(Arrays.asList(arr)); return set.contains(targetValue); } |
使用循环判断
1
2
3
4
5
6
7
|
public static boolean useLoop(String[] arr, String targetValue) { for (String s: arr){ if (s.equals(targetValue)) return true ; } return false ; } |
使用Arrays.binarySearch()
Arrays.binarySearch()方法只能用于有序数组!!!如果数组无序的话得到的结果就会很奇怪。
查找有序数组中是否包含某个值的用法如下:
1
2
3
4
5
6
7
|
public static boolean useArraysBinarySearch(String[] arr, String targetValue) { int a = Arrays.binarySearch(arr, targetValue); if (a > 0 ) return true ; else return false ; } |
时间复杂度
下面的代码可以大概的得出各种方法的时间成本。基本思想就是从数组中查找某个值,数组的大小分别是5、1k、10k。这种方法得到的结果可能并不精确,但是是最简单清晰的方式。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
|
public static void main(String[] args) { String[] arr = new String[] { "CD" , "BC" , "EF" , "DE" , "AB" }; //use list long startTime = System.nanoTime(); for ( int i = 0 ; i < 100000 ; i++) { useList(arr, "A" ); } long endTime = System.nanoTime(); long duration = endTime - startTime; System.out.println( "useList: " + duration / 1000000 ); //use set startTime = System.nanoTime(); for ( int i = 0 ; i < 100000 ; i++) { useSet(arr, "A" ); } endTime = System.nanoTime(); duration = endTime - startTime; System.out.println( "useSet: " + duration / 1000000 ); //use loop startTime = System.nanoTime(); for ( int i = 0 ; i < 100000 ; i++) { useLoop(arr, "A" ); } endTime = System.nanoTime(); duration = endTime - startTime; System.out.println( "useLoop: " + duration / 1000000 ); //use Arrays.binarySearch() startTime = System.nanoTime(); for ( int i = 0 ; i < 100000 ; i++) { useArraysBinarySearch(arr, "A" ); } endTime = System.nanoTime(); duration = endTime - startTime; System.out.println( "useArrayBinary: " + duration / 1000000 ); } |
运行结果:
1
2
3
4
|
useList: 13 useSet: 72 useLoop: 5 useArraysBinarySearch: 9 |
使用一个长度为1k的数组
1
2
3
4
5
6
|
String[] arr = new String[ 1000 ]; Random s = new Random(); for ( int i= 0 ; i< 1000 ; i++){ arr[i] = String.valueOf(s.nextInt()); } |
结果:
1
2
3
4
|
useList: 112 useSet: 2055 useLoop: 99 useArrayBinary: 12 |
使用一个长度为10k的数组
1
2
3
4
5
6
|
String[] arr = new String[ 10000 ]; Random s = new Random(); for ( int i= 0 ; i< 10000 ; i++){ arr[i] = String.valueOf(s.nextInt()); } |
结果:
1
2
3
4
|
useList: 1590 useSet: 23819 useLoop: 1526 useArrayBinary: 12 |
总结
显然,使用一个简单的循环方法比使用任何集合都更加高效。许多开发人员为了方便,都使用第一种方法,但是他的效率也相对较低。因为将数组压入Collection类型中,首先要将数组元素遍历一遍,然后再使用集合类做其他操作。
如果使用Arrays.binarySearch()
方法,数组必须是已排序的。由于上面的数组并没有进行排序,所以该方法不可使用。
实际上,如果你需要借助数组或者集合类高效地检查数组中是否包含特定值,一个已排序的列表或树可以做到时间复杂度为O(log(n)),hashset可以达到O(1)。
(英文原文结束,以下是译者注)
使用ArrayUtils
除了以上几种以外,Apache Commons类库中还提供了一个ArrayUtils
类,可以使用其contains
方法判断数组和值的关系。
1
2
3
4
|
import org.apache.commons.lang3.ArrayUtils; public static boolean useArrayUtils(String[] arr, String targetValue) { return ArrayUtils.contains(arr,targetValue); } |
同样使用以上几种长度的数组进行测试,得出的结果是该方法的效率介于使用集合和使用循环判断之间(有的时候结果甚至比使用循环要理想)。
1
2
3
4
5
6
7
8
9
10
11
|
useList: 323 useSet: 3028 useLoop: 141 useArrayBinary: 12 useArrayUtils: 181 ------- useList: 3703 useSet: 35183 useLoop: 3218 useArrayBinary: 14 useArrayUtils: 3125 |
其实,如果查看ArrayUtils.contains
的源码可以发现,他判断一个元素是否包含在数组中其实也是使用循环判断的方式。
部分代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
if (array == null ) { return - 1 ; } else { if (startIndex < 0 ) { startIndex = 0 ; } int i; if (objectToFind == null ) { for (i = startIndex; i < array.length; ++i) { if (array[i] == null ) { return i; } } } else if (array.getClass().getComponentType().isInstance(objectToFind)) { for (i = startIndex; i < array.length; ++i) { if (objectToFind.equals(array[i])) { return i; } } } return - 1 ; } |
所以,相比较之下,我更倾向于使用ArrayUtils工具类来进行一些合数祖相关的操作。毕竟他可以让我少写很多代码(因为自己写代码难免有Bug,毕竟apache提供的开源工具类库都是经过无数开发者考验过的),而且,效率上也并不低太多。
在Java中如何高效的判断数组中是否包含某个元素的更多相关文章
- java中如何高效的判断数组中是否包含某个元素---
package zaLearnpackage; import org.apache.commons.lang3.ArrayUtils; import java.util.Arrays; import ...
- 在Java中怎样高效的推断数组中是否包括某个元素
来自 http://www.hollischuang.com/archives/1269? 怎样检查一个数组(无序)是否包括一个特定的值?这是一个在Java中经经常使用到的并且非常实用的操作.同一时候 ...
- Jquery判断数组中是否包含某个元素$.inArray()的用法
判断数组里面是否包含某个元素可以使用 $.inArray("元素(字符串)",数组名称) 进行判断 ,当存在该元素(字符串)时,返回该元素在数组的下标,不存在时返回 -1 示例代码 ...
- 161101、在Java中如何高效判断数组中是否包含某个元素
如何检查一个数组(无序)是否包含一个特定的值?这是一个在Java中经常用到的并且非常有用的操作.同时,这个问题在Stack Overflow中也是一个非常热门的问题.在投票比较高的几个答案中给出了几种 ...
- 在Java中如何高效判断数组中是否包含某个元素
如何检查一个数组(无序)是否包含一个特定的值?这是一个在Java中经常用到的并且非常有用的操作.同时,这个问题在Stack Overflow中也是一个非常热门的问题.在投票比较高的几个答案中给出了几种 ...
- 《剑指Offer》第1题(Java实现):在一个二维数组中(每个一维数组的长度相同),每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序。请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数。
一.题目描述 在一个二维数组中(每个一维数组的长度相同),每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序.请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该 ...
- JS判断数组中是否有重复元素的方法
判断数组中是否有重复元素,最容易想到的方法是使用2重循环,逐个遍历,比较,但是这个是最慢,最笨的方法,百度得出了更好的方法. var ary = new Array("111",& ...
- js 判断数组中是否存在
/* 判断数组中是否存在 var somearray = ["mon", "tue", "wed", "thur"] s ...
- 面试题:给定一个长度为N的数组,其中每个元素的取值范围都是1到N。判断数组中是否有重复的数字
题目:给定一个长度为N的数组,其中每个元素的取值范围都是1到N.判断数组中是否有重复的数字.(原数组不必保留) 方法1.对数组进行排序(快速,堆),然后比较相邻的元素是否相同.时间复杂度为O(nlog ...
随机推荐
- 数论细节梳理&模板
初阶 扩展欧拉 \(k\ge\varphi(m)\)时,\(b^k\equiv b^{k\%\varphi(m)+\varphi(m)}(\bmod m\)) 扩展CRT 推式子合并同余方程. htt ...
- 【BZOJ2242】计算器(BSGS,快速幂)
[BZOJ2242]计算器(BSGS,快速幂) 题面 BZOJ 洛谷 1.给定y.z.p,计算y^z mod p 的值: 2.给定y.z.p,计算满足xy ≡z(mod p)的最小非负整数x: 3.给 ...
- [luogu3157][bzoj3295][CQOI2011]动态逆序对【cdq分治+树状数组】
题目描述 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一个排列,按照某种顺序依次删除m个元素,你的任务是在每次删除一个元素之前统计整个序列的逆序 ...
- Codeforces 1079D Barcelonian Distance(计算几何)
题目链接:Barcelonian Distance 题意:给定方格坐标,方格坐标上有两个点A,B和一条直线.规定:直线上沿直线走,否则沿方格走.求A到B的最短距离. 题解:通过直线到达的:A.B两点都 ...
- 【docker】docker安装和使用
一.docker简介: docker是容器技术的一个代表,而容器技术是将程序打包和隔离的一种技术,其实它并不是一个新技术,之前在linux内核中早已存在,真正被大众所用所了解是因为docker的出现. ...
- angular2路由与express路由冲突的问题
angular2的路由定义了一个/a,如果走angular的路由没问题,如果直接访问/a就会出现cannot GET /a的错误,原因就是express的路由问题. 所以路由走angular2,那ex ...
- java == 与 equals
1.基本数据类型用"==" java的基本数据类型,也称为原始的数据类型.它们分别是: byte, short, char, int, long, float, double, b ...
- WebAPI接口安全校验
通过网上查看相关WebAPI接口验证的方法,整理了一下,直接上代码,功能不复杂,有问题留言, //--------------------------------------------------- ...
- 关于 vscode 格式化自己的代码 使用shift+alt+f
关于 vscode 格式化自己的代码 使用shift+alt+f,这样就好了
- mfc01
1.解决不能将参数1从“const char []”转换为“LPCTSTR” ,使用多字节字符集.