2016-06-15 11 views
0

Ich habe Klasse Tier mit Feld: Gewicht und Farbe. Wie ich Collections.binarySearch in diesem Fall verwendet wird, kann (verwenden Sie binäre Suche ein Tier durch erforderliche Größe zu finden):Wie verwende ich binarySearch in Sammlungen?

public static int searchElement(final List<? extends Animal> list, final int weight) { 
    return Collections.binarySearch(list, weight...); 
} 
+1

Und bitte vergessen Sie nicht, dass die [ 'list' sortiert werden müssen] (https://docs.oracle.com/javase/8 /docs/api/java/util/Collections.html#binarySearch-java.util.List-T-). – Turing85

+1

Und Animal muss [Comparable] (http://docs.oracle.com/javase/8/docs/api/java/lang/Comparable.html#compareTo%28T%29) implementieren, wenn Sie Ihre Liste sortieren möchten. –

+1

Sie ... kann nicht, wirklich, nicht leicht. 'Collections.binarySearch' würde nur funktionieren, wenn Sie ein Tier mit diesem Gewicht hätten, aber Sie können nicht direkt nach dem Gewicht suchen. Wenn Sie Bibliotheken von Drittanbietern verwenden können, würde Guavas 'Lists.transform' vielleicht helfen. –

Antwort

1

Leider ist es nicht ohne weiteres möglich ist, basierend auf einer bestimmten Eigenschaft für ein Element zu suchen, Verwenden der integrierten Funktionen.

Es gibt mindestens drei Möglichkeiten, wie diese gelöst werden könnte:

  • eine „Vorlage“ mit der gewünschten Eigenschaft anlegen, und die Suche nach dieser
  • die Eigenschaftswerte in einem Array extrahieren, und die Suche in dieses Array
  • eine eigene, eigenschaftsbasierte binäre Suche

Die erste nicht in allen Fällen anwendbar sein kann, und sieht in mancher Hinsicht fragwürdig.

Die zweite ist ziemlich einfach und könnte eine praktikable Option sein. Aber vorausgesetzt, dass Sie eine binäre Suche ausführen, weil die Sammlung große ist, kann dies einige Overhead in Bezug auf Speicher und Leistung auferlegen.

Die dritte Option ist wahrscheinlich die eleganteste und vielseitigste. Zum Glück ist die binarySearch selbst nicht so komplex - nur ein paar Zeilen Code - so ist es einfach, eine eigene zu erstellen, die einige "Schlüssel Extrahieren Function" erhält.

Ich habe diese Ansätze im folgenden Beispiel skizziert:

import java.util.ArrayList; 
import java.util.Arrays; 
import java.util.Collections; 
import java.util.List; 
import java.util.function.Function; 

class Animal implements Comparable<Animal> 
{ 
    private final int weight; 

    Animal(int weight) 
    { 
     this.weight = weight; 
    } 

    public int getWeight() 
    { 
     return weight; 
    } 

    @Override 
    public int compareTo(Animal that) 
    { 
     return Integer.compare(this.weight, that.weight); 
    } 
} 

public class CollectionBinarySearch 
{ 
    public static void main(String[] args) 
    { 
     List<Animal> animals = new ArrayList<Animal>(); 
     animals.add(new Animal(10)); 
     animals.add(new Animal(40)); 
     animals.add(new Animal(20)); 
     animals.add(new Animal(90)); 
     animals.add(new Animal(290)); 
     animals.add(new Animal(130)); 

     Collections.sort(animals); 

     System.out.println(searchWithInstance(animals, 90)); 
     System.out.println(searchWithInstance(animals, 50)); 

     System.out.println(searchWithArray(animals, 90)); 
     System.out.println(searchWithArray(animals, 50)); 

     System.out.println(searchWithFunction(animals, Animal::getWeight, 90)); 
     System.out.println(searchWithFunction(animals, Animal::getWeight, 50)); 

    } 

    public static int searchWithInstance(
     final List<? extends Animal> list, final int weight) { 
     return Collections.binarySearch(list, new Animal(weight)); 
    } 

    public static int searchWithArray(
     final List<? extends Animal> list, final int weight) { 
     int[] array = list.stream().mapToInt(Animal::getWeight).toArray(); 
     return Arrays.binarySearch(array, weight); 
    }   

    // Adapted from Collections#binarySearch 
    private static <T, K extends Comparable<? super K>> int searchWithFunction(
     List<? extends T> list, Function<? super T, K> keyExtractor, K key) { 
     int low = 0; 
     int high = list.size()-1; 
     while (low <= high) { 
      int mid = (low + high) >>> 1; 
      T midVal = list.get(mid); 
      int cmp = keyExtractor.apply(midVal).compareTo(key); 
      if (cmp < 0) 
       low = mid + 1; 
      else if (cmp > 0) 
       high = mid - 1; 
      else 
       return mid; // key found 
     } 
     return -(low + 1); // key not found 
    } 

}