国产一区二区精品-国产一区二区精品久-国产一区二区精品久久-国产一区二区精品久久91-免费毛片播放-免费毛片基地

千鋒教育-做有情懷、有良心、有品質(zhì)的職業(yè)教育機(jī)構(gòu)

手機(jī)站
千鋒教育

千鋒學(xué)習(xí)站 | 隨時隨地免費學(xué)

千鋒教育

掃一掃進(jìn)入千鋒手機(jī)站

領(lǐng)取全套視頻
千鋒教育

關(guān)注千鋒學(xué)習(xí)站小程序
隨時隨地免費學(xué)習(xí)課程

當(dāng)前位置:首頁  >  技術(shù)干貨  > 在Java中如何高效判斷數(shù)組中是否包含某個元素

在Java中如何高效判斷數(shù)組中是否包含某個元素

來源:千鋒教育
發(fā)布人:千鋒老師
時間: 2018-12-19 13:24:00 1545197040

  如何檢查一個數(shù)組(無序)是否包含一個特定的值?這是一個在Java中經(jīng)常用到的并且非常有用的操作。同時,這個問題在Stack Overflow中也是一個非常熱門的問題。在投票比較高的幾個答案中給出了幾種不同的方法,但是他們的時間復(fù)雜度也是各不相同的。本文將分析幾種常見用法及其時間成本。

  檢查數(shù)組是否包含某個值的方法

  使用List

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

  return Arrays.asList(arr).contains(targetValue);}

  使用Set

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

  Setset = new HashSet(Arrays.asList(arr));

  return set.contains(targetValue);}

  使用循環(huán)判斷

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

  for(String s: arr){

  if(s.equals(targetValue))

  return true;

  }

  return false;

  使用Arrays.binarySearch()

  Arrays.binarySearch()方法只能用于有序數(shù)組!!!如果數(shù)組無序的話得到的結(jié)果就會很奇怪。

  查找有序數(shù)組中是否包含某個值的用法如下:

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

  int a = Arrays.binarySearch(arr, targetValue);

  if(a > 0)

  return true;

  else

  return false;}

  時間復(fù)雜度

  下面的代碼可以大概的得出各種方法的時間成本?;舅枷刖褪菑臄?shù)組中查找某個值,數(shù)組的大小分別是5、1k、10k。這種方法得到的結(jié)果可能并不精確,但是是最簡單清晰的方式。

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

  運行結(jié)果:

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

  使用一個長度為1k的數(shù)組

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

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

  結(jié)果:

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

  使用一個長度為10k的數(shù)組

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

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

  結(jié)果:

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

  總結(jié)

  顯然,使用一個簡單的循環(huán)方法比使用任何集合都更加高效。許多開發(fā)人員為了方便,都使用第一種方法,但是他的效率也相對較低。因為將數(shù)組壓入Collection類型中,首先要將數(shù)組元素遍歷一遍,然后再使用集合類做其他操作。

  如果使用Arrays.binarySearch()方法,數(shù)組必須是已排序的。由于上面的數(shù)組并沒有進(jìn)行排序,所以該方法不可使用。

  實際上,如果你需要借助數(shù)組或者集合類高效地檢查數(shù)組中是否包含特定值,一個已排序的列表或樹可以做到時間復(fù)雜度為O(log(n)),hashset可以達(dá)到O(1)。

  使用ArrayUtils

  除了以上幾種以外,Apache Commons類庫中還提供了一個ArrayUtils類,可以使用其contains方法判斷數(shù)組和值的關(guān)系。

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

  return ArrayUtils.contains(arr,targetValue);}

  同樣使用以上幾種長度的數(shù)組進(jìn)行測試,得出的結(jié)果是該方法的效率介于使用集合和使用循環(huán)判斷之間(有的時候結(jié)果甚至比使用循環(huán)要理想)。

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

  其實,如果查看ArrayUtils.contains的源碼可以發(fā)現(xiàn),他判斷一個元素是否包含在數(shù)組中其實也是使用循環(huán)判斷的方式。

  部分代碼如下:

  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工具類來進(jìn)行一些合數(shù)祖相關(guān)的操作。畢竟他可以讓我少寫很多代碼(因為自己寫代碼難免有Bug,畢竟apache提供的開源工具類庫都是經(jīng)過無數(shù)開發(fā)者考驗過的),而且,效率上也并不低太多。

tags:
聲明:本站稿件版權(quán)均屬千鋒教育所有,未經(jīng)許可不得擅自轉(zhuǎn)載。
10年以上業(yè)內(nèi)強(qiáng)師集結(jié),手把手帶你蛻變精英
請您保持通訊暢通,專屬學(xué)習(xí)老師24小時內(nèi)將與您1V1溝通
免費領(lǐng)取
今日已有369人領(lǐng)取成功
劉同學(xué) 138****2860 剛剛成功領(lǐng)取
王同學(xué) 131****2015 剛剛成功領(lǐng)取
張同學(xué) 133****4652 剛剛成功領(lǐng)取
李同學(xué) 135****8607 剛剛成功領(lǐng)取
楊同學(xué) 132****5667 剛剛成功領(lǐng)取
岳同學(xué) 134****6652 剛剛成功領(lǐng)取
梁同學(xué) 157****2950 剛剛成功領(lǐng)取
劉同學(xué) 189****1015 剛剛成功領(lǐng)取
張同學(xué) 155****4678 剛剛成功領(lǐng)取
鄒同學(xué) 139****2907 剛剛成功領(lǐng)取
董同學(xué) 138****2867 剛剛成功領(lǐng)取
周同學(xué) 136****3602 剛剛成功領(lǐng)取
相關(guān)推薦HOT
云原生存儲和云存儲有什么區(qū)別?

一、架構(gòu)設(shè)計不同云原生存儲是指在云原生環(huán)境下設(shè)計和構(gòu)建的存儲系統(tǒng)。它是基于云原生計算模式和原則進(jìn)行設(shè)計,充分利用容器、微服務(wù)和自動化管...詳情>>

2023-10-14 06:50:34
如何刪除需要使用管理員權(quán)限才能刪除的文件?

如何刪除需要使用管理員權(quán)限才能刪除的文件在Windows系統(tǒng)中,有時候我們可能會遇到一些需要管理員權(quán)限才能刪除的文件。這是因為這些文件可能是...詳情>>

2023-10-14 06:27:57
有什么好用的redis可視化管理工具?

一、Redis Desk較好 ManagerRedis Desk較好 Manager是一款非常受歡迎的Redis數(shù)據(jù)庫管理工具。它支持直接進(jìn)行數(shù)據(jù)修改、刪除和新增等操作,而且...詳情>>

2023-10-14 06:24:43
市場上C++主要是用來做什么的?

C++是一種廣泛應(yīng)用于市場上的編程語言,具有高性能和強(qiáng)大的功能。它的設(shè)計目標(biāo)是提供高效的底層控制和與硬件交互的能力,同時保持可移植性和可...詳情>>

2023-10-14 06:01:51
為什么說PHP與Swoole是優(yōu)異組合?

PHP與Swoole在Web開發(fā)和網(wǎng)絡(luò)編程領(lǐng)域中,共同構(gòu)成了一種強(qiáng)大的解決方案。什么是PHPPHP是一種廣泛使用的開源服務(wù)器端腳本語言,特別適用于Web開...詳情>>

2023-10-14 05:08:41