怎样高效地去判断Array中是否包含某个值?
问题
怎样去判断Array(无序)中是否包含某个值呢?
这是一个在Java中经常被问到的问题。它也是Stack Overflow上投票前几的一个问题。下面将展示投票前几的几个回答,这些回答使用不同的方式解决了这个问题,但是,时间复杂度也是各有不同的。
四种解决方法
使用List
public static boolean useList(String[] arr, String targetValue) {return Arrays.asList(arr).contains(targetValue);}
使用Set
public static boolean useSet(String[] arr, String targetValue) {Set<String> set = new HashSet<String>(Arrays.asList(arr));return set.contains(targetValue);}
使用Loop
public static boolean useLoop(String[] arr, String targetValue) {for (String s : arr) {if (s.equals(targetValue))return true;}return false;}
使用ArraysBinarySearch,这个方法只适用于已排序的Array
public static boolean useArraysBinarySearch(String[] arr, String targetValue) {int a = Arrays.binarySearch(arr, targetValue);if (a > 0)return true;elsereturn false;}
时间复杂度
这大概的运行时间可以通过下面的代码进行衡量。这基本的思路是通过在大小分别为5,1k,10k的Array中进行查找。这个方法可能不够精确,但是这个思路还是很简单清晰的。
注:测试环境为win7-64+JDK1.6
使用Array大小为5
public static void main(String[] args) {System.out.println("---------array size is 5-----------");String[] arr = new String[] { "CD", "BC", "EF", "DE", "AB" };//use listlong 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 setstartTime = 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 loopstartTime = 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+" 毫秒");}
结果:
---------array size is 5-----------useList: 10 毫秒useSet: 66 毫秒useLoop: 4 毫秒useArrayBinary: 4 毫秒
### 使用Array大小为1k
System.out.println("---------array size is 1k----------");String[] arr = new String[1000];Random s = new Random();for (int i = 0; i < 1000; i++) {arr[i] = String.valueOf(s.nextInt());}
结果:
---------array size is 1k----------useList: 762 毫秒useSet: 7549 毫秒useLoop: 677 毫秒useArrayBinary: 14 毫秒
使用Array大小为10k
System.out.println("---------array size is 10k---------");String[] arr = new String[10000];Random s = new Random();for (int i = 0; i < 10000; i++) {arr[i] = String.valueOf(s.nextInt());}
结果:
---------array size is 10k---------useList: 7395 毫秒useSet: 126341 毫秒useLoop: 5776 毫秒useArrayBinary: 17 毫秒
总结
从测试的结果可以清楚地知道,使用简单的Loop比使用集合操作更加有效 。许多开发者使用List方式,但是那并不高效。把Array放到另一个Collection中需要读取Array中的所有元素,这将花费不少的时间。
使用Arrays.binarySearch()的前提是这Array必须是有序的。
实际上,如果你真正需要高效地去判断Array/Collection中是否包含某个值,一个排序了的List或者Tree的时间复杂度为O(log(n)),或者用HashSet,它的时间复杂度为O(1)。
原文链接:http://www.programcreek.com/2014/04/check-if-array-contains-a-value-java/
翻译:crane-yuan
[ 转载请保留原文出处、译者和译文链接。]
怎样高效地去判断Array中是否包含某个值?的更多相关文章
- node js 判断数组中是否包含某个值
判断数组中是否包含某个值这里有四种方法.用的测试数据: let arr=["a","b","c"]; let arr2={"a&q ...
- js判断数组中是否包含某个值
/** * 判断数组中是否包含某个值 * @param arr 数组 * @param str 值 * @returns {boolean} */ function contains(arr, str ...
- Js 判断数组中是否包含某个值
includes() 方法用来判断一个数组是否包含一个指定的值,如果是返回 true,否则false. JavaScript Array includes() 方法
- List containsKey 和Map contains 判断集合中是否包含某个值
map集合 //1.第一种 HashMap map = new HashMap(); map.put("1", "value1"); map.put(" ...
- 161101、在Java中如何高效判断数组中是否包含某个元素
如何检查一个数组(无序)是否包含一个特定的值?这是一个在Java中经常用到的并且非常有用的操作.同时,这个问题在Stack Overflow中也是一个非常热门的问题.在投票比较高的几个答案中给出了几种 ...
- 在Java中如何高效的判断数组中是否包含某个元素
原文出处: hollischuang(@Hollis_Chuang) 如何检查一个数组(无序)是否包含一个特定的值?这是一个在Java中经常用到的并且非常有用的操作.同时,这个问题在Stack Ove ...
- java中如何高效判断数组中是否包含某个特定的值
四种不同方式检查数组是否包含某个值 使用List: public static boolean useList(String[] arr, String targetValue) { return A ...
- 在Java中如何高效判断数组中是否包含某个元素
如何检查一个数组(无序)是否包含一个特定的值?这是一个在Java中经常用到的并且非常有用的操作.同时,这个问题在Stack Overflow中也是一个非常热门的问题.在投票比较高的几个答案中给出了几种 ...
- java中如何高效的判断数组中是否包含某个元素---
package zaLearnpackage; import org.apache.commons.lang3.ArrayUtils; import java.util.Arrays; import ...
随机推荐
- saiku执行速度慢
使用saiku的过程中发现一个重要问题,速度慢!下面是跟踪和优化过程 一.首先抓包,发现ajax请求:http://l-tdata2.tkt.cn6.qunar.com:8080/saiku/rest ...
- 一段时间没上来了,看到有很多网友对OWA感兴趣,因为所在公司发展方向的原因,没有太多时间继续深入研究OWA,敬请见谅
一段时间没上来了,看到有很多网友对OWA感兴趣,因为所在公司发展方向的原因,没有太多时间继续深入研究OWA,敬请见谅
- Xcode 安装插件手误选择了「Skip Bundle」后需要重新允许「Load Bundle」的解决方法
在 Mac 终端输入命令: # 这里的7.1代表 Xcode 的版本号 defaults delete com.apple.dt.Xcode DVTPlugInManagerNonApplePlugI ...
- safenet 超级狗 java调用 小计
最近使用safenet 的 超级狗 在调用api的时候遇到了一些小问题记录一下. 首先safeNetdemo中的api是针对demo的 不能用于部署使用 ,真正用户部署使用的是在用工具包,超级狗开发商 ...
- C中调用Lua函数
我们先来看一个简单的例子: lua_State* L = NULL; // 内部调用lua函数 double f(double x, double y) { double z; lua_getglob ...
- Adroid学习系列-入门(1)
1. 安装 安装Eclipse,安装Adroid插件.安装Adroid SDK. 2. 项目目录说明 2.1. 建立Adroid项目 与一般的Java项目一样 )src文件夹是项目的所 ...
- 基于 CentOS7 的 Kubernetes 集群
一.环境 相关主机 master 192.168.12.197 minion 192.168.12.198~232 etcd 192.168.12.233~236 相关软件 OS CentOS 7 软 ...
- android程序---->android多线程下载(一)
多线程下载是加快下载速度的一种方式,通过开启多个线程去执行一个任务,可以使任务的执行速度变快.多线程的任务下载时常都会使用得到断点续传下载,就是我们在一次下载未结束时退出下载,第二次下载时会接着第一次 ...
- 缓存池扩展 (Buffer Pool Extension)实践
SQL Server 2014缓存池扩展 (Buffer Pool Extension)功能可以将缓存池扩展到较快的SSD存储上.为内存比较紧张的系统提供了新的扩展途径. Buffer Pool 扩展 ...
- Java魔法堂:打包知识点之jar
一.前言 通过eclipse导出jar包十分方便快捷,但作为码农岂能满足GUI的便捷呢?所以一起来CLI吧! 二.JAR包 JAR包是基于ZIP文件格式,用于将多个.java文件和各种资源文件, ...