java - 使用泛型 Java 对数组/比较器进行二分搜索

标签 java arrays generics search binary

我正在努力使一个用于二分搜索的比较器在对象数组上工作。本质上,目标是搜索不规则数组以找到项目的第一个匹配项或最接近的匹配项以提供插入点。该方法传入一个泛型(这是不可更改的 - 因为这是家庭作业),但您无法创建泛型类型的数组...因此,我的比较器抛出错误:“数组类型中的方法二元搜索(对象[],对象)不适用于参数(对象[],E,比较器)”。也许我需要转换通用元素“item”?我不知道。代码:

private Location findFirst(E item) {
    Location current;
    int closestMatchArray1;
    int closestMatchArray2;

Object[] firstItemInArray2 = new Object[numArrayInUse];
Object firstItem;

Comparator<E> comparator = new CompareElement();

for (int i - 0; i < numArrayInUse; i++) {
    firstItem = topArray[i];
    firstItemInArray2[i] = firstItem;
}

closestMatchArray1 = Arrays.binarySearch(firstItemInArray2, item, comparator);

次要但相关的问题。在比较器中,我尝试调用 Comparable 方法“compareTo”来获取一个负整数,该负整数给出了搜索失败时某个项目在数组中的大致位置,但同样,我在泛型方面遇到了麻烦,收到此错误:“方法compareTo(E) 对于类型 E 是未定义的”。代码:

public class CompareElement implements Comparator<E> {
  public int compare(E firstItem, E secondItem) {
     return firstItem.compareTo(secondItem);
  }
}

最佳答案

我认为您要么需要 Comparator<Object>或者你需要一个 E[] 的数组。对于后者,我建议查看这两篇文章:

鼓励阅读以上内容。

<小时/>

选项 1:Array.newInstance()

假设 item 永远不为 null

根据我在上述帖子中读到的内容,如果您知道 item 永远不会是 null ,你可以尝试这样的事情......

@SuppressWarnings("unchecked")
private Location findFirst(E item) {
    Location current;
    int closestMatchArray1;
    int closestMatchArray2;

    // Object[] firstItemInArray2 = new Object[numArrayInUse];
    // Object firstItem;

    E[] firstItemInArray2 
            = (E[]) Array.newInstance(item.getClass(), numArrayInUse); 
    E firstItem;

    Comparator<E> comparator = new CompareElement();

    for (int i = 0; i < numArrayInUse; i++) {
        firstItem = (E) topArray[i];
        firstItemInArray2[i] = firstItem;
    }

    closestMatchArray1 = Arrays.binarySearch(firstItemInArray2, item, comparator);
<小时/>

选项2:Array.newInstance()

需要类参数

如果您不能保证该商品永远不会 null ,并且您无法为 null 提供任何特殊处理值,您可以强制 Class<?>传入的参数,如下...

@SuppressWarnings("unchecked")
private Location findFirst(E item, Class<E> clazz) {
    Location current;
    int closestMatchArray1;
    int closestMatchArray2;

    // Object[] firstItemInArray2 = new Object[numArrayInUse];
    // Object firstItem;

    E[] firstItemInArray2 
            = (E[]) Array.newInstance(clazz, numArrayInUse); 
    E firstItem;

    Comparator<E> comparator = new CompareElement();

    for (int i = 0; i < numArrayInUse; i++) {
        firstItem = (E) topArray[i];
        firstItemInArray2[i] = firstItem;
    }

    closestMatchArray1 = Arrays.binarySearch(firstItemInArray2, item, comparator);
<小时/>

选项 3:对象比较器包装

丑陋,但有效

或者,您可以创建 Comparator<Object>包装您现有的Comparator<E> ,如下(我认为这有点黑客,但它对我来说始终有效)...

    private Location findFirst(E item) {
    Location current;
    int closestMatchArray1;
    int closestMatchArray2;

    Object[] firstItemInArray2 = new Object[numArrayInUse];
    Object firstItem;

    // Comparator<E> comparator = new CompareElement();
    Comparator<Object> comparator = new Comparator<Object>() {
        private final Comparator<E> delegate = new CompareElement();

        @Override
        @SuppressWarnings("unchecked")
        public int compare(Object o1, Object o2) {
            return delegate.compare((E) o1, (E) o2);
        }
    };

    for (int i = 0; i < numArrayInUse; i++) {
        firstItem = topArray[i];
        firstItemInArray2[i] = firstItem;
    }

    closestMatchArray1 = Arrays.binarySearch(firstItemInArray2, item, comparator);
<小时/>

希望这有帮助!

关于java - 使用泛型 Java 对数组/比较器进行二分搜索,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12839947/

相关文章:

c# - 从随机数组中提取最小值和最大值以显示

java - 如何推断泛型类型以避免未经检查的转换

generics - 如何在 Javadoc 链接中使用泛型类型参数?

java - 数组和 if 语句 : Array object not passable into if statement

java - 从 HTTP XML 响应中提取元素 - HTTP 客户端和 Java

java - Selenium 测试的屏幕截图尺寸较小

c# - 从泛型类中的接口(interface)继承

java - 我们可以为 Java 中的构造函数提供返回类型吗?

javascript - 为列添加新的摘要行共享 2 个不同列的相同值并处理时间差

javascript - Java 实体数组到 JavaScript 数组