Java中高效判断数组中是否包含某个元素

如何检查一个数组(无序)是否包含一个特定的值?这是一个在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 set = new HashSet(Arrays.asList(arr));

return set.contains(targetValue);}

使用循环判断

public static boolean useLoop(String[] arr, String targetValue) {

for(String s: arr){

if(s.equals(targetValue))

return true;

}

return false;

使用Arrays.binarySearch()

Arrays.binarySearch()方法只能用于有序数组!!!如果数组无序的话得到的结果就会很奇怪。

查找有序数组中是否包含某个值的用法如下:

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。这种方法得到的结果可能并不精确,但是是最简单清晰的方式。

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);}

运行结果:

useList: 13useSet: 72useLoop: 5useArraysBinarySearch: 9

使用一个长度为1k的数组

String[] arr = new String[1000];Random s = new Random();for(int i=0; i< 1000; i++){

arr[i] = String.valueOf(s.nextInt());}

结果:

useList: 112useSet: 2055useLoop: 99useArrayBinary: 12

使用一个长度为10k的数组

String[] arr = new String[10000];Random s = new Random();for(int i=0; i< 10000; i++){

arr[i] = String.valueOf(s.nextInt());}

结果:

useList: 1590useSet: 23819useLoop: 1526useArrayBinary: 12

总结

显然,使用一个简单的循环方法比使用任何集合都更加高效。许多开发人员为了方便,都使用第一种方法,但是他的效率也相对较低。因为将数组压入Collection类型中,首先要将数组元素遍历一遍,然后再使用集合类做其他操作。

如果使用Arrays.binarySearch()方法,数组必须是已排序的。由于上面的数组并没有进行排序,所以该方法不可使用。

实际上,如果你需要借助数组或者集合类高效地检查数组中是否包含特定值,一个已排序的列表或树可以做到时间复杂度为O(log(n)),hashset可以达到O(1)。

使用ArrayUtils

除了以上几种以外,Apache Commons类库中还提供了一个ArrayUtils类,可以使用其contains方法判断数组和值的关系。

import org.apache.commons.lang3.ArrayUtils;public static boolean useArrayUtils(String[] arr, String targetValue) {

return ArrayUtils.contains(arr,targetValue);}

同样使用以上几种长度的数组进行测试,得出的结果是该方法的效率介于使用集合和使用循环判断之间(有的时候结果甚至比使用循环要理想)。

useList: 323useSet: 3028useLoop: 141useArrayBinary: 12useArrayUtils: 181-------useList: 3703useSet: 35183useLoop: 3218useArrayBinary: 14useArrayUtils: 3125

其实,如果查看ArrayUtils.contains的源码可以发现,他判断一个元素是否包含在数组中其实也是使用循环判断的方式。

部分代码如下:

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架构师的道路上披荆斩棘。需要资料的欢迎加入学习交流群:9285,05736

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/503125.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

最常问的Java面试题,有Get吗?

1. import java和javax有什么区别 刚开始的时候JavaAPI所必需的包是java开头的包&#xff0c;javax当时只是扩展API包来说使用。然而随着时间的推移&#xff0c;javax逐渐的扩展成为Java API的组成部分。但是&#xff0c;将扩展从javax包移动到java包将是太麻烦了&#xff0c;…

Java的类型擦除

一、各种语言中的编译器是如何处理泛型的 通常情况下&#xff0c;一个编译器处理泛型有两种方式&#xff1a; 1.Code specialization。在实例化一个泛型类或泛型方法时都产生一份新的目标代码&#xff08;字节码or二进制代码&#xff09;。例如&#xff0c;针对一个泛型list&…

如何高效学习java呢?

1.基础阶段的学习 在这个学习阶段你应该找一个好的基础学习视频&#xff0c;对着视频认真进行学习&#xff0c;每天自己要逼着自己学习&#xff0c;大学很容易让人颓废。边看还要变敲代码&#xff0c;不要用ide用记事本有利于后期的代码编写习惯&#xff0c;还要做好笔记哦。 …

Java中数组的初始化

一、什么是初始化 在Java程序开发中&#xff0c;使用数组之前都会对其进行初始化&#xff0c;这是因为数组是引用类型&#xff0c;声明数组只是声明一个引用类型的变量&#xff0c;并不是数组对象本身&#xff0c;只要让数组变量指向有效的数组对象&#xff0c;程序中就可使用…

Java程序员如何不断提高自己的专业技能

1、一定要看书 现在学习Java变得比以前容易多了&#xff0c;除了有大量的视频教程外&#xff0c;还有专业的java培训机构&#xff0c;这都使学习变得更加傻瓜化&#xff0c;然而我要说的是&#xff0c;Java虽然变得越来越容易学&#xff0c;然而那只代表入门容易&#xff0c;并…

Java学习路线,你值得了解

Java学习路线&#xff0c;了解一下&#xff01; 需要资料的欢迎加入学习交流群&#xff1a;9285&#xff0c;05736

_linux中curl命令详解-linux运维

linux中curl命令的使用详情&#xff1a;1、【curl url】获取该网址的文本信息&#xff1b;2、【curl -i url】获取该网址的文本信息以及协议头部信息&#xff1b;3、【curl -x proxy url】使用代理获取网页文本信息。linux中curl命令的使用详情&#xff1a;1、curl url(获取该网…

Java的重载与覆盖,傻傻分不清!

重载和覆盖是JAVA多态性的不同表现方式&#xff0c;其中重载是在一个类中多态性的一种表现&#xff0c;是指在一个类中定义了多个同名的方法&#xff0c;他们或有不同参数个数或有不同参数类型。 在使用重载时需注意以下几点&#xff1a; 1&#xff09;重载是通过参数来区分的…

python字符串_教你快速了解 Python 字符串

我们知道字符串是 Python 中最常用的数据类型。我们可以使用引号(或")来创建字符串。创建字符串很简单&#xff0c;只要为变量分配一个值即可。例如&#xff1a;var1 Hello World! var2 "Python Runoob"Python 访问字符串中的值Python 不支持单字符类型&#…

那些年踩过的Java异常,简直了!

1、NullPointerException 空指针异常&#xff0c;操作一个 null 对象的方法或属性时会抛出这个异常。具体看这篇文章&#xff1a;Java 避免空指针的 5 个案例。 2、OutOfMemoryError 内存异常异常&#xff0c;这不是程序能控制的&#xff0c;是指要分配的对象的内存超出了当前最…

字体怎么安装到电脑上_文章还在使用电脑上的固定字体?这款字体软件超好用...

在我们平日里的工作以及生活当中&#xff0c;我们经常会看到一些文字的文档。但是&#xff0c;有的时候我们会因为文字在文档内太多单一&#xff0c;从而觉得枯燥乏味一点了都没有。尤其是在用电脑读文本小说、或者是说明文档的的时候&#xff0c;字体千篇一律&#xff0c;看着…

java面向对象多态特性

一、多态的概念 多态是继封装&#xff0c;继承之后&#xff0c;面向对象的三大特性。 现实事物经常会体现出多种形态&#xff0c;如学生&#xff0c;学生是人的一种&#xff0c;则一个具体的张三同学既是学生也是人&#xff0c;即出现两种形态。 java作为面向对象的语言&…

决策树 prepruning_智能建筑运维前探 AI天天见之五:决策树算法应用探索

点击上方“蓝字”关注我们&#xff0c;成为幸福的人~这是一个大街小巷热议人工智能的时代~人工智能犹如上世纪九十年代的互联网&#xff0c;以惊涛之势席卷大江南北、带动着产业革命。作为智慧建筑运维领域的专业服务商&#xff0c;有志于“用数字智能服务每个建筑&#xff0c;…

Java技术知识点的一些总结

边学习边记录&#xff0c;对于程序员来说真的是一个很好的学习习惯&#xff0c;下面是近期在学习java时总结的一些技术知识点。 第一讲&#xff0c;主要是Java环境变量的配置。搭建好环境才能干活。要理解什么是编译、什么是运行。编译是javac.exe程序在负责&#xff1b;运行则…

电脑机器人_【头条】厚积薄发!卡达电脑智能机器人之纸箱码垛机赋能智能工厂...

点击上方视频&#xff0c;观看卡达电脑“智能机器人之纸箱码垛机”现场运行视频。近年来&#xff0c;随着纸包装行业市场竞争加剧、订单交付时间变短、人工成本逐年增加、产品质量要求变高等挑战&#xff0c;使得纸箱包装智能化生产变得极为迫切。“软件硬件” 助力瓦楞纸箱包装…

如何参悟Java的三大框架

大数据时代&#xff0c;中国IT环境也将面临重新洗牌&#xff0c;不仅仅是企业&#xff0c;更是程序员们转型可遇而不可求的机遇。 Java是世界第一编程语言&#xff0c;这已经达成共识&#xff0c;是毋庸置疑的真理。框架是程序员们必学的知识点&#xff0c;而且是十分重要的应用…

amba simple class驱动_UML - 建模基础 - 用例驱动

用例驱动是统一过程的重要概念&#xff0c;或者说整个软件生产过程就是用例驱动。用例驱动软件生产过程是非常有道理的。让我们再次回顾建模公式&#xff0c;很容易得出一个推论&#xff0c;要解决问题领域就要归纳出所有必要的抽象角度(用例)&#xff0c;为这些用例描述出可能…

ad file type not recognised_Java实用工具类:File工具类方法学习,可创建目录及文件...

开发项目过程中&#xff0c;会用到很多工具类&#xff0c;今天分享一个Java中File操作工具类&#xff0c;可以帮你节约时间&#xff0c;提高开发效率。File工具类public class CreateFileUtil {public static boolean createFile(String destFileName) { File file new File(d…

putty保存用户名和密码_社保局官网搬家啦!忘记用户名、密码肿么办?自救指南来一波!...

刚过完元旦四舍五入一下马上又要到春节了又开始期盼着下一个假期新年换新衣新年新气象我们官网也换新网址啦&#xff01;宝宝们还记得前几期的内容吗&#xff1f;小编再三提醒过大家一定要收藏新网址办理社保业务不要跑错地方哦&#xff01;深圳市社会保险基金管理局官网已经开…

hibernate 批量更新_360产品动态,10月份平台都有哪些更新?

本篇给大家分享的是360资源上下线产品变化快讯&#xff0c;下面一起来看下10月份360平台都有哪些产品更新。每月初定期更新上月内容哦~1产品优化| 新功能上线▌展示广告10月13日 创意质量上线概述&#xff1a;展示广告-投放管理-创意层级-诊断&#xff0c;修改为“创意质量”。…